亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

JavaScript實現區塊鏈

2019-11-19 14:10:33
字體:
來源:轉載
供稿:網友

幾乎每個人都聽說過像比特幣和以太幣這樣的加密貨幣,但是只有極少數人懂得隱藏在它們背后的技術。在這篇文中,我將會用JavaScript來創建一個簡單的區塊鏈來演示它們的內部究竟是如何工作的。我將會稱之為SavjeeCoin!

全文分為三個部分:

  1. part1:實現一個基本的區塊鏈
  2. part2:實現POW
  3. part3:交易與挖礦獎勵

Part1:實現一個基本的區塊鏈

區塊鏈

區塊鏈是由一個個任何人都可以訪問的區塊構成的公共數據庫。這好像沒什么特別的,不過它們有一個有趣的屬性:它們是不可變的。一旦一個區塊被添加到區塊鏈中,除非讓剩余的其余區塊失效,否則它是不會再被改變的。

這就是為什么加密貨幣是基于區塊鏈的原因。你肯定不希望人們在交易完成后再變更交易!

創造一個區塊

區塊鏈是由許許多多的區塊鏈接在一起的(這聽上去好像沒毛病..)。鏈上的區塊通過某種方式允許我們檢測到是否有人操縱了之前的任何區塊。

那么我們如何確保數據的完整性呢?每個區塊都包含一個基于其內容計算出來的hash。同時也包含了前一個區塊的hash。

下面是一個區塊類用JavaScript寫出來大致的樣子:

const SHA256 = require("crypto-js/sha256");class Block { constructor(index, timestamp, data, previousHash = '') { this.index = index; this.previousHash = previousHash; this.timestamp = timestamp; this.data = data; this.hash = this.calculateHash(); } calculateHash() { return SHA256(this.index + this.previousHash + this.timestamp + JSON.stringify(this.data)).toString(); }}

因為JavaScript中并不支持sha256所以我引入了 crypto-js 庫。然后我定義了一個構造函數來初始化我區塊的屬性。每一個區塊上都被賦予了 index 屬性來告知我們這個區塊在整個鏈上的位置。我們同時也生成了一個時間戳,以及需要在區塊里存儲的一些數據。最后是前一個區塊的hash。

創造一個鏈

現在我們可以在Blockchain類中將區塊鏈接起來了!下面是用JavaScript實現的代碼:

class Blockchain{ constructor() { this.chain = [this.createGenesisBlock()]; } createGenesisBlock() { return new Block(0, "01/01/2017", "Genesis block", "0"); } getLatestBlock() { return this.chain[this.chain.length - 1]; } addBlock(newBlock) { newBlock.previousHash = this.getLatestBlock().hash; newBlock.hash = newBlock.calculateHash(); this.chain.push(newBlock); } isChainValid() { for (let i = 1; i < this.chain.length; i++){  const currentBlock = this.chain[i];  const previousBlock = this.chain[i - 1];  if (currentBlock.hash !== currentBlock.calculateHash()) {  return false;  }  if (currentBlock.previousHash !== previousBlock.hash) {  return false;  } } return true; }}

在構造函數里,我通過創建一個包含創世塊的數組來初始化整個鏈。第一個區塊是特殊的,因為它不能指向前一個區塊。我還添加了下面兩個方法:

  • getLatestBlock() 返回我們區塊鏈上最新的區塊。
  • addBlock() 負責將新的區塊添加到我們的鏈上。為此,我們將前一個區塊的hash添加到我們新的區塊中。這樣我們就可以保持整個鏈的完整性。因為只要我們變更了最新區塊的內容,我們就需要重新計算它的hash。當計算完成后,我將把這個區塊推進鏈里(一個數組)。

最后,我創建一個 isChainValid() 來確保沒有人篡改過區塊鏈。它會遍歷所有的區塊來檢查每個區塊的hash是否正確。它會通過比較 previousHash 來檢查每個區塊是否指向正確的上一個區塊。如果一切都沒有問題它會返回 true 否則會返回 false 。

使用區塊鏈

我們的區塊鏈類已經寫完啦,可以真正的開始使用它了!

let savjeeCoin = new Blockchain();savjeeCoin.addBlock(new Block(1, "20/07/2017", { amount: 4 }));savjeeCoin.addBlock(new Block(2, "20/07/2017", { amount: 8 }));

在這里我僅僅是創建了一個區塊鏈的實例,并且命名它為SavjeeCoin!之后我在鏈上添加了一些區塊。區塊里可以包含任何你想要放的數據,不過在上面的代碼里,我選擇添加了一個帶有 amount 屬性的對象。

試著操作吧!

在介紹里我曾說過區塊鏈是不可變的。一旦添加,區塊就不可能再變更了。讓我們試一下!

// 檢查是否有效(將會返回true)console.log('Blockchain valid? ' + savjeeCoin.isChainValid());// 現在嘗試操作變更數據savjeeCoin.chain[1].data = { amount: 100 };// 再次檢查是否有效 (將會返回false)console.log("Blockchain valid? " + savjeeCoin.isChainValid());

我會在一開始通過運行 isChainValid() 來驗證整個鏈的完整性。我們操作過任何區塊,所以它會返回true。

之后我將鏈上的第一個(索引為1)區塊的數據進行了變更。之后我再次檢查整個鏈的完整性,發現它返回了false。我們的整個鏈不再有效了。

結論

這個小栗子還遠未達到完成的程度。它還沒有實現POW(工作量證明機制)或P2P網絡來與其它礦工來進行交流。

但他確實證明了區塊鏈的工作原理。許多人認為原理會非常復雜,但這篇文章證明了區塊鏈的基本概念是非常容易理解和實現的。

Part2:實現POW(proof-of-work:工作量證明)

在part1中我們用JavaScript創建了一個簡單的區塊鏈來演示區塊鏈的工作原理。不過這個實現并不完整,很多人發現依舊可以篡改該系統。沒錯!我們的區塊鏈需要另一種機制來抵御攻擊。那么讓我們來看看我們該如何做到這一點!

問題

現在我們可以很快的創造區塊然后非常迅速的將它們添加進我們的區塊鏈中。不過這導致了三個問題:

  • 第一:人們可以快速創建區塊然后在我們的鏈里塞滿垃圾。大量的區塊會導致我們區塊鏈過載并讓其無法使用。
  • 第二:因為創建一個有效的區塊太容易了,人們可以篡改鏈中的某一個區塊,然后重新計算所有區塊的hash。即使它們已經篡改了區塊,他們仍然可以以有效的區塊來作為結束。
  • 第三:你可以通過結合上述兩個破綻來有效控制區塊鏈。區塊鏈由p2p網絡驅動,其中節點會將區塊添加到可用的最長鏈中。所以你可以篡改區塊,然后計算所有其他的區塊,最后添加多任意你想要添加的區塊。你最后會得到一個最長的鏈,所有的其它節點都會接受它然后往上添加自己的區塊。

顯然我們需要一個方案來解決這些問題:POW。

什么是POW

POW是在第一個區塊鏈被創造之前就已經存在的一種機制。這是一項簡單的技術,通過一定數量的計算來防止濫用。工作量是防止垃圾填充和篡改的關鍵。如果它需要大量的算力,那么填充垃圾就不再值得。

比特幣通過要求hash以特定0的數目來實現POW。這也被稱之為 難度

不過等一下!一個區塊的hash怎么可以改變呢?在比特幣的場景下,一個區塊包含有各種金融交易信息。我們肯定不希望為了獲取正確的hash而混淆了那些數據。

為了解決這個問題,區塊鏈添加了一個 nonce 值。Nonce是用來查找一個有效Hash的次數。而且,因為無法預測hash函數的輸出,因此在獲得滿足難度條件的hash之前,只能大量組合嘗試。尋找到一個有效的hash(創建一個新的區塊)在圈內稱之為挖礦。

在比特幣的場景下,POW確保每10分鐘只能添加一個區塊。你可以想象垃圾填充者需要多大的算力來創造一個新區塊,他們很難欺騙網絡,更不要說篡改整個鏈。

實現POW

我們該如何實現呢?我們先來修改我們區塊類并在其構造函數中添加Nonce變量。我會初始化它并將其值設置為0。

constructor(index, timestamp, data, previousHash = '') { this.index = index; this.previousHash = previousHash; this.timestamp = timestamp; this.data = data; this.hash = this.calculateHash(); this.nonce = 0;}

我們還需要一個新的方法來增加Nonce,直到我們獲得一個有效hash。強調一下,這是由難度決定的。所以我們會收到作為參數的難度。

mineBlock(difficulty) { while (this.hash.substring(0, difficulty) !== Array(difficulty + 1).join("0")) {  this.nonce++;  this.hash = this.calculateHash(); } console.log("BLOCK MINED: " + this.hash);}

最后,我們還需要更改一下 calculateHash() 函數。因為目前他還沒有使用Nonce來計算hash。

calculateHash() { return SHA256(this.index + this.previousHash + this.timestamp + JSON.stringify(this.data) + this.nonce ).toString();}

將它們結合在一起,你會得到如下所示的區塊類:

class Block { constructor(index, timestamp, data, previousHash = '') { this.index = index; this.previousHash = previousHash; this.timestamp = timestamp; this.data = data; this.hash = this.calculateHash(); this.nonce = 0; } calculateHash() { return SHA256(this.index + this.previousHash + this.timestamp + JSON.stringify(this.data) + this.nonce).toString(); } mineBlock(difficulty) { while (this.hash.substring(0, difficulty) !== Array(difficulty + 1).join("0")) {  this.nonce++;  this.hash = this.calculateHash(); } console.log("BLOCK MINED: " + this.hash); }}

修改區塊鏈

現在,我們的區塊已經擁有Nonce并且可以被開采了,我們還需要確保我們的區塊鏈支持這種新的行為。讓我們先在區塊鏈中添加一個新的屬性來跟蹤整條鏈的難度。我會將它設置為2(這意味著區塊的hash必須以2個0開頭)。

constructor() { this.chain = [this.createGenesisBlock()]; this.difficulty = 2;}

現在剩下要做的就是改變 addBlock() 方法,以便在將其添加到鏈中之前確保實際挖到該區塊。下面我們將難度傳給區塊。

addBlock(newBlock) { newBlock.previousHash = this.getLatestBlock().hash; newBlock.mineBlock(this.difficulty); this.chain.push(newBlock);}


大功告成!我們的區塊鏈現在擁有了POW來抵御攻擊了。

測試

現在讓我們來測試一下我們的區塊鏈,看看在POW下添加一個新區塊會有什么效果。我將會使用之前的代碼。我們將創建一個新的區塊鏈實例然后往里添加2個區塊。

let savjeeCoin = new Blockchain();console.log('Mining block 1');savjeeCoin.addBlock(new Block(1, "20/07/2017", { amount: 4 }));console.log('Mining block 2');savjeeCoin.addBlock(new Block(2, "20/07/2017", { amount: 8 }));

如果你運行了上面的代碼,你會發現添加新區塊依舊非??臁_@是因為目前的難度只有2(或者你的電腦性能非常好)。

如果你創建了一個難度為5的區塊鏈實例,你會發現你的電腦會花費大概十秒鐘來挖礦。隨著難度的提升,你的防御攻擊的保護程度越高。

免責聲明

就像之前說的:這絕不是一個完整的區塊鏈。它仍然缺少很多功能(像P2P網路)。這只是為了說明區塊鏈的工作原理。

并且:由于單線程的原因,用JavaScript來挖礦并不快。

Part3:交易與挖礦獎勵

在前面兩部分我們創建了一個簡單的區塊鏈,并且加入了POW來抵御攻擊。然而我們在途中也偷了懶:我們的區塊鏈只能在一個區塊中存儲一筆交易,而且礦工沒有獎勵?,F在,讓我們解決這個問題!

重構區塊類

現在一個區塊擁有 index , previousHash , timestamp , data , hash 和 nonce 屬性。這個 index 屬性并不是很有用,事實上我甚至不知道為什么開始我要將它添加進去。所以我把它移除了,同時將 data 改名為 transactions 來更語義化。

class Block{ constructor(timestamp, transactions, previousHash = '') { this.previousHash = previousHash; this.timestamp = timestamp; this.transactions = transactions; this.hash = this.calculateHash(); this.nonce = 0; }}

當我們改變區塊類時,我們也必須更改 calculateHash() 函數?,F在它還在使用老舊的 index 和 data 屬性。

calculateHash() { return SHA256(this.previousHash + this.timestamp + JSON.stringify(this.transactions) + this.nonce).toString();}

交易類

在區塊內,我們將可以存儲多筆交易。因此我們還需要定義一個交易類,一邊我們可以鎖定交易應當具有的屬性:

class Transaction{ constructor(fromAddress, toAddress, amount){ this.fromAddress = fromAddress; this.toAddress = toAddress; this.amount = amount; }}

這個交易例子非常的簡單,僅僅包含了發起方( fromAddress )和接受方( toAddress )以及數量。如果有需求,你也可以在里面加入更多字段,不過這個只是為了最小實現。

調整我們的區塊鏈

當前的最大任務:調整我們的區塊鏈來適應這些新變化。我們需要做的第一件事就是存儲待處理交易的地方。

正如你所知道的,由于POW,區塊鏈可以穩定的創建區塊。在比特幣的場景下,難度被設置成大約每10分鐘創建一個新區塊。但是,是可以在創造兩個區塊之間提交新的交易。

為了做到這一點,首先需要改變我們區塊鏈的構造函數,以便他可以存儲待處理的交易。我們還將創造一個新的屬性,用于定義礦工獲得多少錢作為獎勵:

class Blockchain{ constructor() {  this.chain = [this.createGenesisBlock()];  this.difficulty = 5;  // 在區塊產生之間存儲交易的地方  this.pendingTransactions = [];  // 挖礦回報  this.miningReward = 100; }}

下一步,我們將調整我們的 addBlock() 方法。不過我的調整是指刪掉并重寫它!我們將不再允許人們直接為鏈上添加區塊。相反,他們必須將交易添加至下一個區塊中。而且我們將 addBlock() 更名為 createTransaction() ,這看起來更語義化:

createTransaction(transaction) { // 這里應該有一些校驗! // 推入待處理交易數組 this.pendingTransactions.push(transaction);}

挖礦

人們現在可以將新的交易添加到待處理交易的列表中。但無論如何,我們需要將他們清理掉并移入實際的區塊中。為此,我們來創建一個 minePendingTransactions() 方法。這個方法不僅會挖掘所有待交易的新區塊,而且還會向采礦者發送獎勵。

minePendingTransactions(miningRewardAddress) { // 用所有待交易來創建新的區塊并且開挖.. let block = new Block(Date.now(), this.pendingTransactions); block.mineBlock(this.difficulty); // 將新挖的看礦加入到鏈上 this.chain.push(block); // 重置待處理交易列表并且發送獎勵 this.pendingTransactions = [   new Transaction(null, miningRewardAddress, this.miningReward) ];}

請注意,該方法采用了參數 miningRewardAddress 。如果你開始挖礦,你可以將你的錢包地址傳遞給此方法。一旦成功挖到礦,系統將創建一個新的交易來給你挖礦獎勵(在這個栗子里是100枚幣)。

有一點需要注意的是,在這個栗子中,我們將所有待處理交易一并添加到一個區塊中。但實際上,由于區塊的大小是有限制的,所以這是行不通的。在比特幣里,一個區塊的大小大概是2Mb。如果有更多的交易能夠擠進一個區塊,那么礦工可以選擇哪些交易達成哪些交易不達成(通常情況下費用更高的交易容易獲勝)。

地址的余額

在測試我們的代碼錢讓我們再做一件事!如果能夠檢查我們區塊鏈上地址的余額將會更好。

getBalanceOfAddress(address){ let balance = 0; // you start at zero! // 遍歷每個區塊以及每個區塊內的交易 for(const block of this.chain){  for(const trans of block.transactions){   // 如果地址是發起方 -> 減少余額   if(trans.fromAddress === address){    balance -= trans.amount;   }   // 如果地址是接收方 -> 增加余額   if(trans.toAddress === address){    balance += trans.amount;   }  } } return balance;}

測試

好吧,我們已經完成并可以最終一切是否可以正常工作!為此,我們創建了一些交易:

let savjeeCoin = new Blockchain();console.log('Creating some transactions...');savjeeCoin.createTransaction(new Transaction('address1', 'address2', 100));savjeeCoin.createTransaction(new Transaction('address2', 'address1', 50));

這些交易目前都處于等待狀態,為了讓他們得到證實,我們必須開始挖礦:

console.log('Starting the miner...');savjeeCoin.minePendingTransactions('xaviers-address');

當我們開始挖礦,我們也會傳遞一個我們想要獲得挖礦獎勵的地址。在這種情況下,我的地址是 xaviers-address (非常復雜?。?。

之后,讓我們檢查一下 xaviers-address 的賬戶余額:

console.log('Balance of Xaviers address is', savjeeCoin.getBalanceOfAddress('xaviers-address'));// 輸出: 0

我的賬戶輸出竟然是0?!等等,為什么?難道我不應該得到我的挖礦獎勵么?那么,如果你仔細觀察代碼,你會看到系統會創建一個交易,然后將您的挖礦獎勵添加為新的待處理交易。這筆交易將會包含在下一個區塊中。所以如果我們再次開始挖礦,我們將收到我們的100枚硬幣獎勵!

console.log('Starting the miner again!');savjeeCoin.minePendingTransactions("xaviers-address");console.log('Balance of Xaviers address is', savjeeCoin.getBalanceOfAddress('xaviers-address'));// 輸出: 100

局限性與結論

現在我們的區塊鏈已經可以在一個區塊上存儲多筆交易,并且可以為礦工帶來回報。

不過,還是有一些不足:發送貨幣是,我們不檢查發起人是否有足夠的余額來實際進行交易。然而,這其實是一件容易解決的事情。我們也沒有創建一個新的錢包和簽名交易(傳統上用公鑰/私鑰加密完成)。

免責聲明 & 源代碼

我想指出的是,這絕不是一個完整的區塊鏈實現!它仍然缺少很多功能。這只是為了驗證一些概念來幫助您來了解區塊鏈的工作原理。

該項目的源代碼就放在我的 GitHub

總結

以上所述是小編給大家介紹的JavaScript實現區塊鏈,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對武林網網站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
91黑丝高跟在线| 国产免费一区二区三区在线能观看| 国产成人精品一区二区三区| 欧美黄色成人网| 亚洲级视频在线观看免费1级| 欧美日韩国产一区中文午夜| 5566成人精品视频免费| 亚洲欧美另类在线观看| 久久影视电视剧凤归四时歌| 国产成+人+综合+亚洲欧美丁香花| 中文字幕成人在线| 国产精品男女猛烈高潮激情| 日韩av在线网站| 国产精品久久久久av免费| 在线成人一区二区| 久久国产一区二区三区| 45www国产精品网站| 91丨九色丨国产在线| 欧美激情网友自拍| 国产a∨精品一区二区三区不卡| 日本欧美在线视频| 欧美激情第一页xxx| 91九色视频在线| 91在线无精精品一区二区| 国产精品吹潮在线观看| 97欧美精品一区二区三区| 91精品久久久久久久| 欧美日韩在线另类| 精品成人69xx.xyz| 欧美天堂在线观看| 懂色aⅴ精品一区二区三区蜜月| 欧美麻豆久久久久久中文| 国产精品一区二区三区在线播放| 精品偷拍一区二区三区在线看| 成人福利在线视频| 精品久久香蕉国产线看观看亚洲| 久久久精品免费| 日韩欧美主播在线| 一本色道久久88精品综合| 国产香蕉一区二区三区在线视频| 97久久伊人激情网| 国产精品你懂得| 欧美成人免费一级人片100| 亚洲第一页自拍| 精品久久中文字幕久久av| 欧美激情欧美狂野欧美精品| 亚洲视频在线观看视频| 亚洲精品久久久久久下一站| 国产成人激情小视频| 成人精品视频99在线观看免费| 久久精品小视频| 欧美成人精品在线视频| 久久精品中文字幕一区| 91精品国产亚洲| 色婷婷亚洲mv天堂mv在影片| 欧美日韩国产一区中文午夜| 国产丝袜一区二区| 日韩电影免费观看中文字幕| 成人黄在线观看| 国产黑人绿帽在线第一区| 成人精品福利视频| 国产精品青青在线观看爽香蕉| 成人激情免费在线| 国产亚洲精品一区二555| 亚洲精品按摩视频| 91国产一区在线| 亚洲永久在线观看| 理论片在线不卡免费观看| 欧美激情va永久在线播放| 国产日韩欧美另类| 国产一区视频在线| 成人午夜在线影院| 91tv亚洲精品香蕉国产一区7ujn| 韩剧1988在线观看免费完整版| 国产精品免费网站| 精品国内自产拍在线观看| 欧美激情网友自拍| 日韩电影免费观看在线观看| 91久久久久久| 日韩成人av在线播放| 日韩va亚洲va欧洲va国产| 国产亚洲a∨片在线观看| 亚洲欧美色婷婷| 亚洲国产成人精品一区二区| 亚洲综合中文字幕在线观看| 精品成人久久av| 国产视频在线一区二区| 亚洲欧洲成视频免费观看| 久久久久久久久久久免费精品| 中国日韩欧美久久久久久久久| 国产日产欧美精品| 91精品久久久久久久久久久久久| 亚洲韩国欧洲国产日产av| 色婷婷久久一区二区| 国内精品一区二区三区四区| 日韩美女视频免费在线观看| 成人激情视频小说免费下载| …久久精品99久久香蕉国产| www国产亚洲精品久久网站| 国产视频精品一区二区三区| 成人福利网站在线观看11| 久久久久久久久久国产精品| 久久69精品久久久久久国产越南| 色99之美女主播在线视频| 91在线观看免费观看| 这里只有精品视频在线| 久久亚洲精品小早川怜子66| 激情懂色av一区av二区av| 亚洲毛茸茸少妇高潮呻吟| 亚洲精品美女在线| 亚洲日韩中文字幕在线播放| 久久久久久999| 国产精品久久久久久久久久久久久| 午夜精品国产精品大乳美女| 欧美日韩亚洲成人| 日韩美女免费观看| 日韩精品在线视频美女| 亚洲第一综合天堂另类专| 久久电影一区二区| 日韩av综合中文字幕| 日韩欧美成人区| 成人精品一区二区三区电影免费| 欧美在线视频a| 国产精品久久久久久亚洲影视| 97视频网站入口| 欧美日韩性视频在线| 亚洲欧美中文日韩在线v日本| 精品视频中文字幕| 在线观看91久久久久久| 蜜月aⅴ免费一区二区三区| 欧美精品videofree1080p| 最近中文字幕日韩精品| 亚洲日本欧美日韩高观看| 日韩美女毛茸茸| 亚洲一区二区黄| 日韩在线观看免费网站| 久久国产加勒比精品无码| 欧美高清视频在线播放| 精品人伦一区二区三区蜜桃免费| 性夜试看影院91社区| 91精品中文在线| 久久99青青精品免费观看| 国内精品久久久久| 国产99久久久欧美黑人| 日韩av中文字幕在线免费观看| 亚洲欧美中文在线视频| 成人久久精品视频| 久久男人av资源网站| 黑人与娇小精品av专区| 国产欧美一区二区三区久久人妖| 国产成人福利视频| 久久久久久有精品国产| 91精品久久久久久综合乱菊| 亚洲一区二区三区xxx视频| 欧美大胆a视频| 国产视频精品在线| 亚洲石原莉奈一区二区在线观看| 亚洲精品免费一区二区三区| 亚洲性线免费观看视频成熟| 国外成人在线直播| 91精品国产91久久久久久| 成人国产在线激情| 亚洲成人国产精品| 韩国视频理论视频久久|