【譯註】:來自 Hacker School 的 Mary Rose Cook 實現了一個純 JavaScript 寫就的 Git:Gitlet,包含了最主要的一些命令。這個專案一是為了瞭解 Git 內部原理,二是希望寫一篇深入淺出解釋 Git 核心概念的短文。學習一件東西的原理最好的方法就是去親自實現它,而設計精巧的 Git 核心功能程式碼也不過 300 行。這就是這篇精巧的小文:Git in 600 words,相應的程式碼在 Github 上。短文很有趣,思路清晰也足夠深入,值得一看。
設想你現在位於 alpha/
目錄下,這裡有一個文字檔案 number.txt
,裡面的內容只有一個詞:“first”。
現在執行 git init
將這個 alpha 資料夾初始化為 Git 倉庫。
執行 git add number.txt
會將 number.txt
新增到 Git 的索引(index)中。這個索引記錄了所有 Git 保持追蹤的檔案,現在它有了一個對映記錄 number.txt -> first
,同時 add
命令還會把一個包含了 first
字串的二進位制物件加入 Git 的物件資料庫裡。
現在執行 git commit -m first
。這條命令會做三件事情。首先在物件資料庫內建立一個樹物件,用以記錄 alpha
目錄下的檔案列表,這個物件有一個指標指向前面 git add
命令建立的 first
二進位制物件;第二,這條命令還會建立一個 commit 物件用以代表剛剛提交的版本,它包含一個指標指向剛剛的樹物件;第三,master 分支也會指向這個新建立的 commit 物件。
現在執行 git clone . ../beta
。它會建立一個新目錄 beta 並將其初始化為 Git 倉庫,然後把 alpha 倉庫的物件資料庫中所有物件拷貝給 beta 的物件資料庫,將 beta 的 master 分支像 alpha 的 master 一樣指向相應的物件。它還根據 first
提交的內容配置索引,並根據索引更新目錄下的檔案——也就是 number.txt
。
現在切換到 beta 目錄,修改 number.txt
的內容為“second”,執行 git add number.txt
和 git commit -m second
,新建立的提交物件 second(譯註:姑且稱之為 second)會有一個指向父提交(first)的指標,表示 second 繼承自 first,而 master 分支則指向 second 提交。
回到 alpha 目錄,執行 git remote add beta ../beta
,將 beta 倉庫設為遠端倉庫。然後執行 git pull beta master
。
在這條命令背後,它其實會執行 git fetch beta master
,從 beta 倉庫中找到 second 提交的相關物件拷貝到 alpha 倉庫;把 alpha 中關於 beta 的 master 分支記錄指向這個 second 提交;更新 FETCH_HEAD
指向剛剛從 beta 倉庫拉取的 master 分支,還是這個 second 提交。
此外,pull
命令還會執行 git merge FETCH_HEAD
。從 FETCH_HEAD
得知最近拉取的分支是 beta 倉庫的 master 分支,據此拿到相應的物件,也就是 second 提交物件。此時 alpha 的 master 分支指著 first 提交,正好是 second 的祖先提交,於是對於 merge 命令來說只需要將 master 分支指向 second 提交即可。接下來 merge
命令還會更新索引以匹配 second 提交的內容,並且相應更新工作目錄中的檔案。
現在執行 git branch red
,建立一個名為“red”、指向 second 提交的新分支。
然後執行 git checkout red
。在 checkout 之前,HEAD
指向 master 分支,執行命令之後它就指向了 red 分支,使得 red 成為當前分支。
接下來把 number.txt
的內容修改為 “third”,執行 git add numbers.txt
和 run git commit -m third
。
之後再執行 git push beta red
,這條命令會把 alpha 倉庫內跟 third 提交相關的物件拷貝至 beta 倉庫,並且將(alpha 倉庫內記錄的)beta 倉庫 red 分支指向 third 提交。就醬。