git的commit值是怎么计算出来的

众所周知,代码版本管理工具git会为每一个版本提交创建一个commit值。这个值是一个SHA-1哈希,那么这个值是怎么计算出来的呢?往下看。

我们来看一个仓库的最后一次commit hash值:

$ git show
commit 19d02d2cc358e59b3d04f82677dbf3808ae4fc40 (HEAD -> master, origin/master, origin/HEAD)
Author: Evan Liu <hmisty@gmail.com>
Date:   Tue Jan 16 13:46:12 2018 +0800

    go fmt

字符串19d02d2cc358e59b3d04f82677dbf3808ae4fc40就是最后一次提交的commit hash。这个hash大概是由下面的信息计算出来的:

$ git cat-file commit HEAD
tree df7d111a2509509177674d965b99df8399b8168e
parent b0f8a1eb2a567ec7d163ce9d91d82f676070be2b
author Evan Liu <hmisty@gmail.com> 1516081572 +0800
committer Evan Liu <hmisty@gmail.com> 1516081572 +0800

go fmt

但这还不够,前面要加一个commit NNN,就像这样:

$ printf "commit %s\0" $(git cat-file commit HEAD | wc -c)
commit 209

连起来就是:

$ printf "commit %s\0" $(git cat-file commit HEAD | wc -c); git cat-file commit HEAD
commit 209tree df7d111a2509509177674d965b99df8399b8168e
parent b0f8a1eb2a567ec7d163ce9d91d82f676070be2b
author Evan Liu <hmisty@gmail.com> 1516081572 +0800
committer Evan Liu <hmisty@gmail.com> 1516081572 +0800

go fmt

对上面的输出求SHA-1就可以得到正确的结果了:

$ (printf "commit %s\0" $(git cat-file commit HEAD | wc -c); git cat-file commit HEAD) | shasum
19d02d2cc358e59b3d04f82677dbf3808ae4fc40  -

在你的电脑上求SHA-1的命令可能叫做shasum或者sha1sum。

这正是我们最初git show查看的commit hash。

让我们用git log命令看一下之前版本的commit hash:

commit 19d02d2cc358e59b3d04f82677dbf3808ae4fc40 (HEAD -> master, origin/master, origin/HEAD)
Author: Evan Liu <hmisty@gmail.com>
Date:   Tue Jan 16 13:46:12 2018 +0800

    go fmt

commit b0f8a1eb2a567ec7d163ce9d91d82f676070be2b
Author: Evan Liu <hmisty@gmail.com>
Date:   Tue Jan 16 13:38:33 2018 +0800

    panicIfError

我们回头来看一下作为commit hash计算输入的输入都包含了哪些信息:

tree df7d111a2509509177674d965b99df8399b8168e
parent b0f8a1eb2a567ec7d163ce9d91d82f676070be2b
author Evan Liu <hmisty@gmail.com> 1516081572 +0800
committer Evan Liu <hmisty@gmail.com> 1516081572 +0800

go fmt

第一行,tree哈希,根据这个哈希可以展开本次commit的整个目录树及每个对象的hash值。如果感兴趣,可以用 git cat-file -p 哈希值 来查看到底都是些什么对象:

$ git cat-file -p df7d111a2509509177674d965b99df8399b8168e
100644 blob 5520cde3cf15000bbe6b67442aab61cfbe0cf2d8    .gitignore
100644 blob 3e8d19f93f71e7a062e6009d6719f16cab3e2f30    DEV_GUIDE.md
100644 blob c3a0aa959a84f2df7c109caa1c75627bd5a9bb63    LICENSE
100644 blob 9efb0fcbae6204c32395dcbf63089ad04649fc66    README.md
100644 blob 60a1bf18843435f1b3fd2f016d1785469ffcd4d6    RELEASE_NOTES
100644 blob a33deeeb9b407f59fd447a708cf5dcc3edb78b27    SPEC.md
040000 tree 7b1d8214bd1384a5e48e3d17e8a12922ef52b505    bson_rpc
040000 tree 06602aa29e5a662aaa34b6a7b1b1530185652efa    examples
100644 blob f7a7f93c52486f321619222c0611864ce1631803    requirements.txt
100644 blob 0e579f6e256123a752ebaa332d488c546d4a07c5    setup.py

这里面所有的哈希都可以继续用git cat-file -p继续展开,一直到叶子节点(文件)。

第二行,parent哈希,正是上一个commit的hash值。

第三行,author,时间戳1516081572是2018/1/16 13:46:12,本次提交的时间。

第四行,committer,时间戳仍然是本次提交的时间。

剩余部分是提交时所写的notes。

所以,我们大概可以在脑海中构想出git提交的一个链条结构:

...
|
V
commit hash3
. tree hash ---------> tree of hashes of objects (Merkle Tree)
. parent hash
|
V
commit hash2
. tree hash ---------> tree of hashes of objects (Merkle Tree)
. parent hash
|
V
commit hash1
. tree hash ----------> tree of hashes of objects (Merkle Tree)
. parent hash
|
V
创世哈希(0000000000000000000000000000000000000000)

每一次新版本的提交,git commit都会让这个链条的哈希高度增加一层,用git log可以看到这个大楼。

可以看到,git的每次commit hash都是包含上一次commit hash以及本次修正的文件结构的Merkle Tree的Merkle Root,有没有觉得和block chain区块链的设计思想有异曲同工之处呢?只不过,git的哈希链是一个DAG(Directed Acyclic Graph,有向无环图),而区块链的哈希链是一个由随机抢答产生的哈希数连接起来的链。

QY 20180117

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