一、Hash算法的身影
可以看到,在生成比特幣地址(《精通比特幣》第4章提到),以及生成區塊唯一標識(《精通比特幣》第7章提到):區塊Hash值時(即挖礦的過程),都使用了Hash算法,特別是SHA256算法。比特幣系統本身也就是加密算法的衍生物。
二、SHA256算法過程
本文主要簡述一下SHA256算法的過程,下一篇會大概說一下它的理論基礎。
1. 輸入信息x
SHA256要求報文長度不超過2^64bit。
2. 補位
因為采用的是分組加密的方法,每組長度是512bit,而原始的信息x,不一定是512的倍數,也就是說不一定能剛好分成每組都是512bit的情況。這時候需要補位。
在x后面先補一個1,然后再補多個0,一直到總長度(x以及補的位數)模512是448
3. 補長度
為什么第二步不直接把總長度補到512的倍數呢,因為后面還要補64位,記錄原始信息x的位數。(64 + 448) % 512 = 0。可以看到,只有64位表示原始信息x的位數,所以x的最大長度是2^64。
4. 計算Hash值
前面已經將消息補成了512的倍數,總長度變為512 * N。計算hash值的基本思想是:先將處理完的消息分成N個512bit的數據塊:M[1],M[2],……,M[N],然后一塊一塊的處理:哈希初值H[0]經過第一個數據塊M[1]得到H[1],H[1]經過第二個數據塊M[2]得到H[2],……,依次處理,最后得到H[N]。每個哈希值,比如H[0],由8個32bit組成,32bit又稱為字,也就是說每個hash值都由8個字組成。最后將H[N]的8個字連接成256bit(8 * 32 bit)的最終值。至于為什么可以這么做,我們下一篇再介紹,先看流程。
1) 哈希初值H[0]
SHA256算法中用到的哈希初值H[0](共有8個32bit)為:
H[0][0] = 0x6A09E667
H[0][1] = 0xBB67AE85
H[0][2] = 0x3C6EF372
H[0][3] = 0xA54FF53A
H[0][4] = 0x510E527F
H[0][5] = 0x9B05688C
H[0][6] = 0x1F83D9AB
H[0][7] = 0x5BE0CD19
0x表示的是十六進制,十六進制中一位表示4bit。
注:這些初值是對自然數中前8個質數2、3、5、7、11等的平方根的小數部分取前32bit而來。
2) 循環計算
# M的下標從1開始,M即為M[1], ..., M[N]
for i in range(1, N + 1):
# 第一步,先根據16個32bit的原始消息,生成64個32bit
W = [0] * 64
for t in range(0, 64):
if t < 16:
W[t] = M[i][t]
else:
W[t] = SSIG1(W[t - 2]) + W[t-7] + SSIG0(W[t-15]) + W[t-16]
# 第二步,將上次的hash值,也就是8個32bit,賦值到8個臨時變量,用這8個臨時變量計算
# 原本的hash值在本輪結束的時候仍然需要使用
a = H[i - 1][0]
b = H[i - 1][1]
c = H[i - 1][2]
d = H[i - 1][3]
e = H[i - 1][4]
f = H[i - 1][5]
g = H[i - 1][6]
h = H[i - 1][7]
# 第三步,執行散列計算,內循環是64輪,使用了上面的64個W,以及提前定義好的64個K
for t in range(0, 64):
t1 = h + BSIG1(e) + CH(e, f, g) + K[t] + W[t]
t2 = BSIG0(a) + MAJ(a,b,c)
h = g
g = f
f = e
e = d + t1
d = c
c = b
b = a
a = t1 + t2
# 第四步,將本輪經過64個內循環新計算的臨時值和上次的hash值求和,作為本輪最終的hash值
H[i][0] = a + H[i - 1][0]
H[i][1] = b + H[i - 1][1]
H[i][2] = c + H[i - 1][2]
H[i][3] = d + H[i - 1][3]
H[i][4] = e + H[i - 1][4]
H[i][5] = f + H[i - 1][5]
H[i][6] = g + H[i - 1][6]
H[i][7] = h + H[i - 1][7]
# 最終H[N]的8個32bit返回,即為該消息的SHA256結果
3) 函數和常量說明
(1) 64個常量K為:
K = [
0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5,
0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174,
0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da,
0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967,
0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,
0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070,
0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3,
0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2]
注:這些常數的取值是從自然數中前64個質數的立方根的小數部分取前32bit而來的。
(2) 利用到的函數為:
# 本段代碼沒有使用python語法
CH(x, y, z) = (x AND y) XOR ((NOT x) AND z)
MAJ(x, y, z) = (x AND y) XOR (x AND z) XOR (y AND z)
BSIG0(x) = ROTR^2(x) XOR ROTR^13(x) XOR ROTR^22(x)
BSIG1(x) = ROTR^6(x) XOR ROTR^11(x) XOR ROTR^25(x)
SSIG0(x) = ROTR^7(x) XOR ROTR^18(x) XOR SHR^3(x)
SSIG1(x) = ROTR^17(x) XOR ROTR^19(x) XOR SHR^10(x)
其中,除了與或非外,XOR是異或, ROTR^n是循環右移n位,SHR^n是右移n位。
三、總結
上面就是SHA256的計算過程。
-
Hash算法
+關注
關注
0文章
43瀏覽量
7407 -
比特幣
+關注
關注
57文章
7006瀏覽量
140970
原文標題:區塊鏈系列--比特幣 (3):區塊鏈中的Hash算法
文章出處:【微信號:AI_shequ,微信公眾號:人工智能愛好者社區】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論