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(父区块头))计算</br>
  4 时间戳 该区块产生的近似时间,精确到秒的UNIX时间戳,
必须严格大于前11个区块时间的中值</br>,
同时全节点也会拒绝那些超出自己2个小时时间戳的区块</br>
  4 难度目标 该区块工作量算法的难度目标,已经使用特定算法编码
  4 Nonce 为了找到满足难度目标所设定的随机数,
为了解决32位随机数在算力飞升的情况下不够用的问题,</br>规定时间戳和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一下吧!👻👻👻

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

打赏一个呗

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码支持
扫码打赏,你说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦