這是一篇能讓你迅速了解 Git 工作原理的文章,實戰(zhàn)案例解析,相信我,3 分鐘,絕對能夠有收獲!
Git 的本質是一個文件系統(tǒng)(很重要,記住這句話,理解這句話),工作目錄中的所有文件的歷史版本以及提交記錄(commit)都是以文件對象的方式保存在 .git 目錄中的。
我們先來創(chuàng)建一個名為 git-demo 空目錄,并采用 git init 命令初始化 Git 倉庫。該命令會在工作目錄下生成一個 .git 目錄,該目錄將用于保存工作區(qū)中所有的文件歷史的歷史版本,commit,branch,tag 等所有信息。
$ mkdir git-demo$ cd git-demo$ git init
其目錄結構如下:
圖片
待會我們重點關注下這幾個目錄:
目前 Objects 目錄中還沒有任何內容,我們創(chuàng)建一個文件并提交:
$ git:(master) echo "my project" > README$ git:(master) mkdir src$ git:(master) echo "hello world" > src/file1.txt
添加并提交:
$ git:(master) git add .$ git:(master) git commit -m "init commit"
圖片
從打印輸出可以看到,上面的命令創(chuàng)建了一個 commit 對象,該 commit 包含兩個文件。查看 .git/objects 目錄,可以看到該目錄下增加了 5 個子目錄 06,3b, 82, c5, ca,每個子目錄下有一個以一長串字母數(shù)字命令的文件:
圖片
這一大串是什么?
Git Object 目錄中存儲了三種對象:Commit, Tree 和 Blob,Git 會為對象生成一個文件,并根據(jù)文件信息生成一個 SHA-1 哈希值作為文件內容的校驗和,創(chuàng)建以該校驗和前兩個字符為名稱的子目錄,并以 (校驗和) 剩下 38 個字符為文件命名 ,將該文件保存至子目錄下。
可以通過 git cat-file -t 哈希值 命令查看對象類型,通過 git cat-file -p 哈希值 命令查看對象中的內容,哈希值就是目錄名+文件名,在沒有歧義的情況下,命令可以不用輸入整個哈希值,輸入前幾位即可。
我們挨個看下:
065bca(blob):
圖片
3b18e(blob):
圖片
824244(tree):
圖片
c5bc98(commit):
圖片
ca96(tree):
圖片
認真看圖,大家看完也就差不多清楚了 commit、blob、tree 這幾大對象是什么東西了
從 commit 對象(c5bc98)入手,commit 對象中保存了 commit 的作者,commit 的描述信息,簽名信息以及該 commit 中包含哪些 tree 對象和 blob 對象。從上圖可知包含了 tree 對象(ca96)。
可以把 tree 對象看成這次提交相關的所有文件的根目錄,可以看到 ca96 這個 tree 對象中包含了一個 blob 對象(065bca),即 README 文件,以及一個 tree 對象(824244),即 src 目錄。而 blob 對象存儲的就是真正的內容。
這幾個對象的對應關系如下圖所示:
圖片
現(xiàn)在來看下 HEAD 中的內容,前面說過,HEAD 中存儲的是工作目錄當前狀態(tài)對應的 commit:
$ git:(master) cat .git/HEADref: refs/heads/master
$ git:(master) cat .git/refs/heads/masterc5bc98b8990bedd7444da537320559e601eba87b
c5bc98 正是我們最近的這次 commit!
master 是一個分支名,所以分支(branch)的本質是一個指向 commit 的指針
我們切一個新分支 feat/work:
圖片
查看下 refs/heads/master 和 refs/heads/feat/work 中的 commit 值:
圖片
從其內容可以看到,feat/work 這個 branch 并沒有創(chuàng)建任何新的版本文件,和 master 一樣指向了 c5bc98 這個 commit。
從上面的實驗可以看出,一個 branch 其實只是一個 commit 對象的應用,Git 并不會為每個 branch 存儲一份拷貝,因此在 git 中創(chuàng)建 branch 幾乎沒有任何代價。
接下來我們在 feat/work 這個 branch上進行一些修改,然后提交:
$ git:(feat/work) echo "new line" >> src/file1.txt$ git:(feat/work) echo "do nothing" >> License$ git:(feat/work) git add .$ git:(feat/work) git commit -m "some change"
圖片
查看當前的 HEAD:
圖片
可以看到 HEAD 指向了 feat/work 這個 branch,而 feat/work branch則指向了 8a442 這個commit,master branch 指向的 commit 未變化,仍然是 c5bc98。
查看 8a442 這個commit對象的內容:
圖片
可以看到 commit 有一個 parent 字段,指向了前一個 commit c5bc98。還包含了一個 tree 對象(2a9dd):
圖片
可以觀察到,由于 README 沒有變化,還是指向的 065bca 這個blob對象。License 是一個新建的 blob 對象,src 和 file1.txt 則指向了新版本的對象。
增加了這次 commit 后,Git 中各個對象的關系如下圖所示:
圖片
Tag 和 branch 類似,也是指向某個 commit 的指針。不同的是 tag 創(chuàng)建后其指向的 commit 不能變化,而 branch 創(chuàng)建后,其指針會在提交新的 commit 后向前移動。
本文鏈接:http://www.tebozhan.com/showinfo-26-6179-0.html三分鐘上手!一文看懂 Git 的底層工作原理
聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com