其實這個問題困擾我有一段時間,相信也有人和我同樣有這個困擾,網上已有不少這種解釋了,可是要麼就是無圖,要麼就是解釋的很亂,沒太看懂,通過本身對git的使用,加上向同事請教,算是理解了這個問題,因此寫下來分享一下,我儘可能詳細說明git
假設咱們有以下圖一所示倉庫,該倉庫有master和develop兩個分支,且develop是在(3.added merge.txt file)commit處從master拉出來的分支。 工具
假設如今HEAD在(6.added hello.txt file)處,也就是在master分支最近的一次提交處,此時執行git merge develop, 結果以下圖所示。 fetch
工做原理就是:git 會自動根據兩個分支的共同祖先即 (3.added merge.txt file)這個 commit 和兩個分支的最新提交即 (6.added hello.txt file) 和 (5.added test.txt file) 進行一個三方合併,而後將合併中修改的內容生成一個新的 commit,即圖二的(7.Merge branch ‘develop’)。
這是merge的效果,簡單來講就合併兩個分支並生成一個新的提交。spa
那rebase是這麼工做的呢?
假設初始狀態也是圖一所顯示的。兩個分支一個master,一個develop,此時HEAD在(6.added hello.txt file)處,如今執行git rebase develop,結果以下圖三所示。 3d
能夠看見develop分支分出來分叉不見了,下面來解釋一下它的工做原理:
在執行git rebase develop以前,HEAD在(6.added hello.txt file)處,當執行rebase操做時,git 會從兩個分支的共同祖先 (3.added merge.txt file)開始提取 當前分支(此時是master分支)上的修改,即 (6.added hello.txt file)這個commit,再將 master 分支指向 目標分支的最新提交(此時是develop分支)即(5.added test.txt file) 處,而後將剛剛提取的修改應用到這個最新提交後面。若是提取的修改有多個,那git將依次應用到最新的提交後面,以下兩圖所示,圖四爲初始狀態,圖五爲執行rebase後的狀態。 blog
簡單來講,git rebase提取操做有點像git cherry-pick同樣,執行rebase後依次將當前的提交cherry-pick到目標分支上,而後將在原始分支上的已提取的commit刪除。ip
那何時用merge,何時用rebase呢?
再舉個例子:
初始狀態以下圖六所示:
和以前同樣的是,develop分支也是在 (3.added merge.txt file)處從master分支拉取develop分支。不同的是兩個分支各個commit的時間不一樣,以前develop分支的4和5commit在master分支3以後6以前,如今是develop分支的4提交早於master分支的5提交,develop分支的6提交晚於master的5提交早於master的7提交。 it
在上圖狀況下,在master分支的7commit處,執行git merge develop,結果以下圖七所示: ast
執行git rebase develop,結果以下圖八所示: class
還有一點說明的是,在項目中常常使用git pull來拉取代碼,git pull至關因而git fetch + git merge,若是此時運行git pull -r,也就是git pull –rebase,至關於git fetch + git rebase
最後推薦一些git可視化工具,我用的是gitkraken,這些工具功能基本同樣,看我的喜歡好使用