本文转载自微信公众号「区块链研究实验室」,作者链三丰。转载本文请联系区块链研究实验室公众号。
自互联网以来,锁链技术已被某些人称为最有影响力的发明。尽管公众将区块链与投机性加密货币同义解释,但区块链实际上在现代世界中具有不可思议的广泛应用。实际上,加密货币只是区块链领域的一小部分,生产中的许多解决方案都是由私人组织领导来实现的。
随着区块链技术的迅速发展,技术领域的专业人员越来越需要了解区块链的基础知识及其技术影响力。
毕竟,实现自己的区块链实际上并不像听起来那样复杂。本文中,我们将使用2021年最流行的编程语言之一GoLang实现基本的区块链。那么,让我们开始吧!
了解区块链
区块链是名称符合其含义的少数技术之一。我们可以将区块链视为通过哈希相互连接的信息块,哈希是从输入数据生成的加密的固定输出。由于每个块都通过哈希相互引用,因此如果不大幅度更改链的其余部分,就不可能更改链的任何部分。
区块链中的每个块都可以包含几乎任何内容的数据。一个基本框架将包括每个块的过去交易的所有记录。比特币以类似的方式工作,这就是为什么您可以一直跟踪比特币交易直至Satoshi首次进行加密货币交易的原因。
下面我们有三个区块来创建区块链。第一块是创世块。由于之前没有任何内容,因此前一个哈希字段为空。我们仍然使用时间戳记属性和初始事务作为哈希算法的输入。该算法将吐出一大串数字和字母,这些数字和字母代表了创世纪块的哈希值。
转到块2,我们将创世块的哈希值用作块2先前哈希值。这个动作将创世块与第2块联系起来!接下来,我们将时间戳,交易列表和先前的哈希值作为我们的哈希算法的输入。该算法将为我们提供一个新的哈希值来表示块2。
我们将继续重复该过程任意多次,除了区块的有效性以及存储区块链的能力(比特币的区块链约为330 GB)之外,没有任何其他限制。
在Go中创建一个简单的区块链
创建我们的区块链的第一步是定义什么是区块。Go使我们的生活更轻松地创建自定义类型,我们可以Block使用以下代码来定义类型。在这里,该Block结构具有四个字段以匹配我们上面的图。
- type Block struct {
- timestamp time.Time
- transactions []string
- prevHash []byte
- Hash []byte
- }
下一步是创建构造函数的Go版本,以创建新的块。该函数的输入将使用一个字符串数组来表示事务,以及一个字节数组来表示先前制作的块所对应的先前的哈希值。下一个要研究的NewHash()功能是我们下一步将实现的功能。
- func NewBlock(transactions []string, prevHash []byte) *Block {
- currentTime := time.Now()
- return &Block {
- timestamp: currentTime,
- transactions: transactions,
- prevHash: prevHash,
- Hash: NewHash(currentTime, transactions, prevHash),
- }
- }
该NewHash()函数将时间,事务列表和以前的哈希作为我们的输入参数,同时返回一个字节数组来表示新生成的哈希值。在此功能中,我们基本上只是将所有输入都混入一个称为的单个字节数组中input。我们这样做是使用append()功能的附加time参数,prevHash通过转换time为string用...,以追加作为后缀time片的prevHash切片。
然后,我们遍历transactions并将每个个体附加transaction到input数据blob。有趣的语法string(rune(transaction))...只是Go中将其中的每个元素转换transactions为可以附加到的切片的一种方法input。它正在输入垃圾内容,因此,如果您真的想要,请深入研究。
- func NewHash(time time.Time, transactions []string, prevHash []byte) []byte {
- input := append(prevHash, time.String()...)
- for transaction := range transactions {
- input = append(input, string(rune(transaction))...)
- }
- hash := sha256.Sum256(input)
- return hash[:]
- }
最后,我们使用crypto包转到电话sha256.Sum256()与input作为它的参数。这将为我们的所有数据输入提供新的哈希表示。我们返回时hash[:]使用的[:]语法将hash在返回时切成适当的长度。
这实际上是我们开始链接我们的区块链所需的全部。当然,我们希望在程序中看到某种输出,以便在打印时可以使用一些辅助方法:
- func printBlockInformation(block *Block) {
- fmt.Printf("\ttime: %s\n", block.timestamp.String())
- fmt.Printf("\tprevHash: %x\n", block.prevHash)
- fmt.Printf("\tHash: %x\n", block.Hash)
- printTransactions(block)
- }
- func printTransactions(block *Block) {
- fmt.Println("\tTransactions:")
- for i, transaction := range block.transactions {
- fmt.Printf("\t\t%v: %q\n", i, transaction)
- }
- }
现在剩下的就是创建新的事务,块和哈希。我们可以通过我们的主要方法来做到这一点。我们定义了一个字符串数组来记录我们的区块链中的交易。另请注意,我们如何传递一个空字节数组NewBlock()来生成名为的第一个块genesisBlock。
- func main() {
- genesisTransactions := []string{"Izzy sent Will 50 bitcoin", "Will sent Izzy 30 bitcoin"}
- genesisBlock := NewBlock(genesisTransactions, []byte{})
- fmt.Println("--- First Block ---")
- printBlockInformation(genesisBlock)
- block2Transactions := []string{"John sent Izzy 30 bitcoin"}
- block2 := NewBlock(block2Transactions, genesisBlock.Hash)
- fmt.Println("--- Second Block ---")
- printBlockInformation(block2)
- block3Transactions := []string{"Will sent Izzy 45 bitcoin", "Izzy sent Will 10 bitcoin"}
- block3 := NewBlock(block3Transactions, block2.Hash)
- fmt.Println("--- Third Block ---")
- printBlockInformation(block3)
- }
为了创建新的区块,我们将前一个区块的哈希值NewBlock()与对应的交易历史记录一起传递。如果您想在一个地方查看整个程序,则为:
- package main
- import (
- "crypto/sha256"
- "fmt"
- "time"
- )
- type Block struct {
- timestamp time.Time
- transactions []string
- prevHash []byte
- Hash []byte
- }
- func main() {
- genesisTransactions := []string{"Izzy sent Will 50 bitcoin", "Will sent Izzy 30 bitcoin"}
- genesisBlock := NewBlock(genesisTransactions, []byte{})
- fmt.Println("--- First Block ---")
- printBlockInformation(genesisBlock)
- block2Transactions := []string{"John sent Izzy 30 bitcoin"}
- block2 := NewBlock(block2Transactions, genesisBlock.Hash)
- fmt.Println("--- Second Block ---")
- printBlockInformation(block2)
- block3Transactions := []string{"Will sent Izzy 45 bitcoin", "Izzy sent Will 10 bitcoin"}
- block3 := NewBlock(block3Transactions, block2.Hash)
- fmt.Println("--- Third Block ---")
- printBlockInformation(block3)
- }
- func NewBlock(transactions []string, prevHash []byte) *Block {
- currentTime := time.Now()
- return &Block {
- timestamp: currentTime,
- transactions: transactions,
- prevHash: prevHash,
- Hash: NewHash(currentTime, transactions, prevHash),
- }
- }
- func NewHash(time time.Time, transactions []string, prevHash []byte) []byte {
- input := append(prevHash, time.String()...)
- for transaction := range transactions {
- input = append(input, string(rune(transaction))...)
- }
- hash := sha256.Sum256(input)
- return hash[:]
- }
- func printBlockInformation(block *Block) {
- fmt.Printf("\ttime: %s\n", block.timestamp.String())
- fmt.Printf("\tprevHash: %x\n", block.prevHash)
- fmt.Printf("\tHash: %x\n", block.Hash)
- printTransactions(block)
- }
- func printTransactions(block *Block) {
- fmt.Println("\tTransactions:")
- for i, transaction := range block.transactions {
- fmt.Printf("\t\t%v: %q\n", i, transaction)
- }
- }
如果要运行此程序,将得到以下输出:
- $ go run example.go
- --- First Block ---
- time: 2021-04-05 15:12:18.813294 -0600 MDT m=+0.000074939
- prevHash:
- Hash: 43ec51c50d2b9565f221155a29d8b72307247b08eaf6731cca
- Transactions:
- 0: "Izzy sent Will 50 bitcoin"
- 1: "Will sent Izzy 30 bitcoin"
- --- Second Block ---
- time: 2021-04-05 15:12:18.813477 -0600 MDT m=+0.000257244
- prevHash: 43ec51c50d2b9565f221155a29d8b72307247b08eaf6731cca
- Hash: fcce5323a35cb67b45fe75866582db00fd32baeb92aac448c7
- Transactions:
- 0: "John sent Izzy 30 bitcoin"
- --- Third Block ---
- time: 2021-04-05 15:12:18.813488 -0600 MDT m=+0.000269168
- prevHash: fcce5323a35cb67b45fe75866582db00fd32baeb92aac448c7
- Hash: fc1d3eee286970d85812b47c3a5bf016ae8c1de4f86b8ace972ffa
- Transactions:
- 0: "Will sent Izzy 45 bitcoin"
- 1: "Izzy sent Will 10 bitcoin"
过程可能会很粗糙,但这是创建自己的区块链的基础!