Go语言简单实现比特币挖矿原理

区块链基本概念

区块链本质上是一个对等网络(peer-to-peer)的分布式账本数据库
比特币的底层就采用了区块链的技术架构。区块链本身其实是一串链接的数据区块,每相邻区块之间相互连接,其链接指针是采用密码学哈希算法对区块头进行处理所产生的区块头哈希值。每一个区块数据块中记录了一组采用哈希算法组成的树状交易状态信息,这样保证了每个区块内的交易数据不可篡改,区块链里链接的区块也不可篡改

比特币的交易记录会保存在数据区块中,比特币系统中大约每10分钟会产生一个区块,每个数据区块一般包含区块头(Header)和区块体(Body)两部分,如图所示。

1-1.jpg

区块头封装了当前的版本号(Version)、前一区块地址(Prev-Block)、时间戳(Timestamp)、随机数(Nonce)、当前 区块的目标哈希值(Bits)、Merkle数的根值(Merkle-root)等信息。

区块头分析

前80个字节是区块头

{
    "hash": "000000000003ba27aa200b1cecaad478d2b00432346c3f1f3986da1afd33e506",
    "ver": 1,
    "prev_block": "000000000002d01c1fccc21636b607dfd930d31d01c3a62104612a1719011250",
    "mrkl_root": "f3e94742aca4b5ef85488dc37c06c3282295ffec960994b2c0d5ac2a25a95766",
    "time": 1293623863,
    "bits": 453281356,
    "nonce": 274148111
}

字节 字段 说明
4 版本 区块版本号,表示区块遵守的验证规则
32 父区块头哈希值 前一区块的哈希值,使用SHA(SHA256(父区块头))计算
32 Merkle根 该区块中交易的Merkle树根的哈希值,同样采用SHA(SHA256(父区块头))计算
4 时间戳 该区块产生的近似时间,精确到秒的UNIX时间戳,必须严格大于前11个区块时间的中值,同时全节点也会拒绝那些超出自己2个小时时间戳的区块</br>
4 难度目标 该区块工作量算法的难度目标,已经使用特定算法编码
4 Nonce 为了找到满足难度目标所设定的随机数,为了解决32位随机数在算力飞升的情况下不够用的问题,规定时间戳和coinbase交易信息均可更改,以此扩展nonce的位数

挖矿

区块在挖矿过程中产生。所谓的挖矿实际上是穷举随机数的算法,把上个区块的哈希值加上十分钟内的全部交易单打包,再加上一个算计数,算出一个256位的字符串哈希值,输入的随机数Nonce 使哈希值满足一定的条件就获得这个区块的交易记账权。新产生的区块需要快速的广播出去,以便其他的节点对其进行验证,以防止造假。当记账成功的时候,即获得区块奖励,也就是挖到了比特币。

获得比特币

随着随机数(Nonce)的不断变化,就会产生不同的哈希值,当产生的哈希值左面的连续位数的0值个数大于等于当前区块难度值的时候,即碰撞成功,找到随机数,获得记账权,获得比特币。如下图:

1-2.png

这个难度为4,碰撞出的随机数是87471.随着难度的不断提高,碰撞出随机数的概率就越低。目前比特币矿机最大的算力为蚂蚁大陆的S9矿机(14TH/s)。我就有两台哦,不过现在每天的产量只有0.0009个BTC,少的可怜。。。

代码实现

定义block结构体


type block struct {

    ver       int   //版本号
    prev_block string  //父区块的哈希值
    mekl_root  string   //该区块merkle树的哈希值
    time       string   //时间戳
    bits       int      // 难度

}

获取哈希值


func getHashValueStr(nonce int,blc block) string {

//拼接区块版本号,时间戳,父区块,merkle树,随机数为hv字符串
    hv := strconv.Itoa(blc.ver) + blc.time + blc.prev_block+
        blc.mekl_root + strconv.Itoa(nonce)

    first := sha256.New()

    first.Write([]byte(hv))

// first.Sum(nil)返回值为[]byte 类型的数组,
 而哈希值由每位字符16进制字符组成,所以应用fmt.Springtf
 函数将数组转化为单个字节为16进制拼接而成的哈希字符串

    return fmt.Sprintf("%x",first.Sum(nil))

}

检验随机数生产的哈希函数是否符合条件



func mining(hashStr string, diff int) bool {

    var I int

    for i = 0; i < len(hashStr); i++ {

      //如果i位数 的值不为0,不满足条件,跳出循环
        if hashStr[i] != '0' {
            break
        }
    }
    //判断i的值是否大于当前的难度,大于碰撞成功,否则失败
    return i >= diff

}

获取随机数


func getNonce(blc block)int  {

    nonce := 0
    
    //改变nonce的值,如果返回false,nonce++
    
    for !mining(getHashValueStr(nonce,blc), blc.bits) {

        fmt.Println(getHashValueStr(nonce,blc))
        nonce ++

    }
    fmt.Println(getHashValueStr(nonce,blc))
    fmt.Println("出块成功!")

    return nonce


}

主函数


func main() {

    prev_block := "000000000003ba27aa200b1cecaad478d2b00432346c3f1f3986da1afd33e506"

    mrkl_root := "000000000002d01c1fccc21636b607dfd930d31d01c3a62104612a1719011250"

    time := "1293623863"

    bits := 4

    block := block{1, prev_block, mrkl_root, time, bits}

    nc := getNonce(block)

    fmt.Println(nc)

}

运行结果

1-3

总结

代码只是简单的实现了比特币的挖矿原理,实际上可能比这处理的更细致,考虑的更全面一些。
至于挖矿的事情,兵法有云“兵贵神速”,17年比特币大涨,那时候难度还不是很高,去云南四川那边投资矿厂的人都赚的盆满钵满,我就认识了一位17年初挖矿的矿工,最高峰拥有20000台S9,一年赚了将近10亿,投资成本据说是几十万。当然这些也都需要一些眼光,勇气,我只是在现在看过去罢了,至于那个时候谁都不知道比特币能涨这么猛。包括现在我之所以还会继续购置矿机,是因为我同样认为目前比特币还远远未到达它应有的价值,产量虽然少,但是挖出的币我是不会卖的,5年以后,10年以后再看,那可能同样现在看过去的感觉。当然这些不是重点,也不构成投资建议。重点看代码,看代码!😂


github源代码

据说找工作github的Star很重要,哈哈哈。喜欢就star一下吧!👻👻👻

作者原创,转载请注明出处,如有错误描述,请评论纠正,谢谢大家!🐳🐳🐳

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352