圖解Git

林堯彬發表於2020-04-04

再次感謝原著作者和中文翻譯者。

此頁圖解git中的最常用命令。如果你稍微理解git的工作原理,這篇文章能夠讓你理解的更透徹。 如果你想知道這個站點怎樣產生,請前往GitHub repository

正文

  1. 基本用法
  2. 約定
  3. 命令詳解
    1. Diff
    2. Commit
    3. Checkout
    4. Detached HEAD(匿名分支提交)
    5. Reset
    6. Merge
    7. Cherry Pick
    8. Rebase
  4. 技術說明

基本用法

上面的四條命令在工作目錄、暫存目錄(也叫做索引)和倉庫之間複製檔案。

  • git add files把當前檔案放入暫存區域。
  • git commit給暫存區域生成快照並提交。
  • git reset -- files用來撤銷最後一次git add files,你也可以用git reset撤銷所有暫存區域檔案。
  • git checkout -- files把檔案從暫存區域複製到工作目錄,用來丟棄本地修改。

你可以用git reset -p,git checkout -p, orgit add -p進入互動模式。

也可以跳過暫存區域直接從倉庫取出檔案或者直接提交程式碼。

  • git commit -a相當於執行git add把所有當前目錄下的檔案加入暫存區域再執行。git commit.
  • git commit files進行一次包含最後一次提交加上工作目錄中檔案快照的提交。並且檔案被新增到暫存區域。
  • git checkout HEAD -- files回滾到複製最後一次提交。

約定

後文中以下面的形式使用圖片。

綠色的5位字元表示提交的ID,分別指向父節點。分支用橘色顯示,分別指向特定的提交。當前分支由附在其上的HEAD標識。 這張圖片裡顯示最後5次提交,ed489是最新提交。 master分支指向此次提交,另一個maint分支指向祖父提交節點。

命令詳解

Diff

有許多種方法檢視兩次提交之間的變動。下面是一些示例。

Commit

提交時,git用暫存區域的檔案建立一個新的提交,並把此時的節點設為父節點。然後把當前分支指向新的提交節點。下圖中,當前分支是master。 在執行命令之前,master指向ed489,提交後,master指向新的節點f0cec並以ed489作為父節點。

即便當前分支是某次提交的祖父節點,git會同樣操作。下圖中,在master分支的祖父節點maint分支進行一次提交,生成了1800b。 這樣,maint分支就不再是master分支的祖父節點。此時,合併 (或者 衍合) 是必須的。

如果想更改一次提交,使用git commit --amend。git會使用與當前提交相同的父節點進行一次新提交,舊的提交會被取消。

另一個例子是分離HEAD提交,後文講。

Checkout

checkout命令通常用來從倉庫中取出檔案,或者在分支中切換。

checkout命令讓git把檔案複製到工作目錄和暫存區域。比如git checkout HEAD~ foo.c把檔案從foo.c提交節點HEAD~ (當前提交節點的父節點)複製到工作目錄並且生成索引。注意當前分支沒有變化。

如果沒有指定檔名,而是一個本地分支,那麼將切換到那個分支去。同時把索引和工作目錄切換到那個分支對應的狀態。

如果既沒有指定檔名,也沒有指定分支名,而是一個標籤、遠端分支、SHA-1值或者是像master~3類似的東西,就得到一個匿名分支,稱作detached HEAD。 這樣可以很方便的在歷史版本之間互相切換。但是,這樣的提交是完全不同的,詳細的在下面

用分離HEAD提交(找不到好的譯法)

HEAD是分離的時候, 提交可以正常進行, 但是沒有更新已命名的分支. 。(可以看作是匿名分支。)

如果此時切換到別的分支,那麼所作的工作會全部丟失。注意這個命令之後就不存在2eecb了。

如果你想儲存當前的狀態,可以用這個命令建立一個新的分支:git checkout -b name

Reset

reset命令把當前分支指向另一個位置,並且有選擇的變動工作目錄和索引。也用來在從歷史倉庫中複製檔案到索引,而不動工作目錄。

如果不給選項,那麼當前分支指向到那個提交。如果用--hard選項,那麼工作目錄也更新,如果用--soft選項,那麼都不變。

如果沒有給出提交點的版本號,那麼預設用HEAD。這樣,分支指向不變,但是索引會回滾到最後一次提交,如果用--hard選項,工作目錄也同樣。

如果給了檔名(或者-p選項), 那麼工作效果和帶檔名的checkout差不多,除了索引被更新。

Merge

merge 命令把不同分支合併起來。合併前,索引必須和當前提交相同。如果另一個分支是當前提交的祖父節點,那麼合併命令將什麼也不做。 另一種情況是如果當前提交是另一個分支的祖父節點,就導致fast-forward合併。指向只是簡單的移動,並生成一個新的提交。

否則就是一次真正的合併。預設把當前提交(ed489 如下所示)和另一個提交(33104)以及他們的共同祖父節點(b325c)進行一次三方合併。結果是先儲存當前目錄和索引,然後和父節點33104一起做一次新提交。

Cherry Pick

cherry-pick命令"複製"一個提交節點並在當前複製做一次完全一樣的新提交。

Rebase

衍合是合併命令的另一種選擇。合併把兩個父分支合併進行一次提交,提交歷史不是線性的。衍合在當前分支上重演另一個分支的歷史,提交歷史是線性的。 本質上,這是線性化的自動的 cherry-pick

上面的命令都在topic分支中進行,而不是master分支,在master分支上重演,並且把分支指向新的節點。注意舊提交沒有被引用,將被回收。

要限制回滾範圍,使用--onto選項。下面的命令在master分支上重演當前分支從169a6以來的最近幾個提交,即2c33a

同樣有git rebase --interactive讓你更方便的完成一些複雜操組,比如丟棄、重排、修改、合併提交。沒有圖片體現著下,細節看這裡:git-rebase(1)

技術說明

檔案內容並沒有真正儲存在索引(.git/index)或者提交物件中,而是以blob的形式分別儲存在資料庫中(.git/objects),並用SHA-1值來校驗。 索引檔案用識別碼列出相關的blob檔案以及別的資料。對於提交來說,以樹(tree)的形式儲存,同樣用對於的雜湊值識別。樹對應著工作目錄中的資料夾,樹中包含的 樹或者blob物件對應著相應的子目錄和檔案。每次提交都儲存下它的上一級樹的識別碼。

如果用detached HEAD提交,那麼最後一次提交會被the reflog for HEAD引用。但是過一段時間就失效,最終被回收,與git commit --amend或者git rebase很像。

 

url:http://my.oschina.net/xdev/blog/114383

轉載於:https://www.cnblogs.com/hnrainll/archive/2013/03/19/2968449.html

相關文章