git rebase VS git merge?更优雅的Git合并方式值得拥有
写在前面
你在使用Git合并分支时只会使用git merge
吗?有时使用git rebase
可以比git merge
做出更优雅的操作。
另外,如果你不能很好的应用Git,那么这里为你提供一个非常棒的Git在线练习工具:Git Online,你可以更直观的看到你所使用的命令会产生什么效果。
Merge与Rebase
不知怎么,git rebase
命令被赋予了一个神奇的污毒声誉,初学者应该远离它,但它实际上可以让开发团队在使用时更加轻松。
你可以将它理解成「七伤拳」,「七伤拳」并不是不能练,只是练「七伤拳」有一个先诀条件,那就是内功境界一定要非常高,即你要充分理解
git rebase
命令。
在本文中,我们将git rebase
与git merge
命令进行比较。在Git工作流中,说明所有可以使用Rebase的场景。
概念概述
关于git rebase
,首先要理解的是它解决了和git merge
同样的问题。这两个命令都旨在将更改从一个分支合并到另一个分支,但二者的合并方式却有很大的不同。
当你在专用分支上开发新feature时,另一个团队成员在master
分支提交了新的commits,这会发生什么?这会导致分叉的历史记录。对于这个问题,使用Git作为协作工具的任何人来说都应该很熟悉。
现在,假设在master
分支上的新提交与你正在开发的feature相关。需要将新提交合并到你的feature
分支中,你可以有两个选择:merge或者rebase。
Merge方式
最简单的方式是通过以下命令将master
分支合并到feature
分支中:
1 | git checkout feature |
或者,你可以将其浓缩为一行命令:
git merge feature master |
这会在feature
分支中创建一个新的merge commit,它将两个分支的历史联系在一起,请看如下所示的分支结构:
使用merge是很好的方式,因为它是一种非破坏性的操作。现有分支不会以任何方式被更改,这避免了rebase操作所产生的潜在缺陷(下面讨论)。
另一方面,这也意味着feature
分支每次需要合并上游更改时,它都将产生一个额外的合并提交。如果master
提交非常活跃,这可能会严重污染你的feature
分支历史记录。尽管可以使用高级选项git log
缓解此问题,但它可能使其他开发人员难以理解项目的历史记录。
Rebase方式
作为merge的替代方法,你可以使用以下命令将master
分支合并到feature
分支上:
1 | git checkout feature |
这会将整个feature
分支移动到master
分支的顶端,从而有效地整合了所有master
分支上的提交。但是,与merge提交方式不同,rebase通过为原始分支中的每个提交创建全新的commits来重写项目历史记录。
rebase的主要好处是可以获得更清晰的项目历史。首先,它消除了git merge
所需的不必要的合并提交;其次,正如你在上图中所看到的,rebase会产生完美线性的项目历史记录,你可以在feature
分支上没有任何分叉的情况下一直追寻到项目的初始提交。这样可以通过命令git log
,git bisect
和gitk
更容易地导航查看项目。
但是,针对这样的提交历史我们需要权衡其「安全性」和「可追溯性」。如果你不遵循Rebase的黄金法则,重写项目历史记录可能会对你的协作工作流程造成灾难性后果。而且,rebase会丢失合并提交的上下文,你也就无法看到上游更改是何时合并到feature中的。
交互式Rebase
交互式rebase使你有机会在将commits移动到新分支时更改这些commits。这比自动rebase更强大,因为它提供了对分支提交历史的完全控制。通常,这用于在合并feature
分支到 master
之前清理其杂乱的历史记录。
要使用交互式rebase,需要使用git rebase
和-i
选项:
1 | git checkout feature |
这将打开一个文本编辑器,列出即将移动的所有提交:
1 | pick 33d5b7a Message for commit #1 |
此列表准确定义了执行rebase后分支的外观。通过更改pick
命令或重新排序条目,你可以使分支的历史记录看起来像你想要的任何内容。例如,如果第二次提交fix了第一次提交中的一个小问题,您可以使用以下fixup
命令将它们浓缩为一个提交:
1 | pick 33d5b7a Message for commit #1 |
保存并关闭文件时,Git将根据您的指示执行rebase,从而产生如下所示的项目历史记录:
消除这种无意义的提交使你的功能历史更容易理解。这是git merge
根本无法做到的事情。至于commits条目前的pick
、fixup
、squash
等命令,在Git目录执行git rebase -i
即可查看到,大家按需重排或合并提交即可,注释说明非常清晰,在此不做过多说明。
Rebase的黄金法则
一旦你理解了什么是rebase,最重要的是要学习什么时候不能使用它。git rebase
的黄金法则是永远不要在公共分支上使用它。
例如,想想如果你rebasemaster
分支到feature
分支之上会发生什么:
rebase将所有master
分支上的提交移动 feature 分支的顶端。问题是这只发生在你自己的存储库中。所有其他开发人员仍在使用原始版本的master
。由于rebase产生了全新 commit,Git会认为你的master
分支历史与其他人的历史不同。
此时,同步两个master
分支的唯一方法是将它们合并在一起,但是这样会产生额外的合并提交和两组包含相同更改的提交(原始提交和通过rebase更改的分支提交)。不用说,这是一个令人非常困惑的情况。
因此,在你运行git rebase
命令之前,总是问自己,还有其他人在用这个分支吗? 如果答案是肯定的,那就把你的手从键盘上移开,开始考虑采用非破坏性的方式进行改变(例如git revert
命令)。否则,你可以随心所欲地重写历史记录。
Force Push
如果你尝试将rebase了的master
分支推送回remote repository,Git将阻止你这样做,因为它会与远程master
分支冲突。但是,你可以通过传递--force
标志来强制推送,如下所示:
1 | # Be very careful with this command! |
这样你自己repository的内容将覆盖远程master
分支的内容,但这会使团队的其他成员感到困惑。因此,只有在确切知道自己在做什么时才要非常小心地使用此命令。
如果没有人在feature分支上作出更改,你可以使用force push将本地内容推送到remote repository做清理工作。
工作流程演练
rebase可以根据你所在团队的需要方便的整合到现有的Git工作流程中。在本节中,我们将了解rebase在功能开发的各个阶段可以提供的好处。
在任何工作流程中,利用git rebase
是为每个功能创建专用分支。这为你提供了必要的分支,以安全地利用rebase:
本地清理
将rebase纳入工作流程的最佳方法之一是清理本地正在进行的功能。通过定期执行交互式rebase,你可以确保功能中的每个提交都具有针对性和意义。这可以使你在编写代码时无需担心将其分解为隔离的提交(多个提交),你可以在事后修复整合它。
使用git rebase
时,有两种情况:feature
父分支(例如master
)的提交,或在feature
中的早期提交。我们在交互式 Rebase部分已经介绍了第一种情况的示例。我们来看后一种情况,当你只需要修复最后几次提交时,以下命令仅做最后3
次提交的交互式rebase。
1 | git checkout feature |
通过指定HEAD~3
,你实际上并没有移动分支,你只是交互式地重写其后的3
个提交。请注意,这不会将上游更改合并到feature
分支中。
如果要使用此方法重写整个功能,git merge-base
命令可用于查找feature
分支的原始base。以下内容返回原始base的提交ID,然后你可以将其传递给git rebase
:
git merge-base feature master |
交互式rebase的使用是引入git rebase
工作流的好方法,因为它只影响本地分支。其他开发人员唯一能看到的就是你提交的最终版,这应该是一个简洁易懂易跟踪的分支历史记录。
但同样,这仅适用于私有feature
分支。如果你通过相同的功能分支(公共分支)与其他开发人员协作,那么你是不被允许重写其历史记录的。
将上游更改合并到功能分支中
在概念概述部分中,我们了解了feature
分支可以使用git merge
或git rebase
合并master
分支的上游更改。merge是一个安全的方式,可以保留git repository的整个历史记录,而rebase则是通过将feature
分支移动到master
顶端来创建线性历史记录。
这种使用git rebase
类似于本地清理,但在此过程中它包含了那些来自master上游的提交。
请记住,将当前提交rebase到远程branch(非master分支)一样是合法的。当与另一个开发人员协作使用相同的功能并且你需要将他们的更改合并到你的repository时,就会发生这种情况。
例如,如果你和另一个名为John的开发人员添加了对feature
分支的提交,在你fetch(注意fetch并不会自动merge)来自John的远程feature
分支后,你的repository可能如下所示:
你可以合并来自上游的分叉:要么用john/feature
merge本地feature
,要么rebase本地feature
到john/feature
的顶部。
请注意,此rebase不违反Rebase 黄金规则,因为只有你的本地feature
提交被移动,之前的所有内容都不会受到影响。这就像是说「将我的更改添加到John已经完成的工作中」。在大多数情况下,这比通过合并提交与远程分支同步更直观。
默认情况下,使用git pull
命令执行合并,但你可以通过向其传递--rebase
选项来强制它将远程分支以rebase方式集成。
git pull --rebase |
使用Pull请求Review Feature
如果你在代码审查过程中使用pull请求,在使用了pull请求之后你应该避免使用git rebase
。一旦你发出pull请求,其他开发人员就会查看你的提交,这意味着它是一个公共分支。重写其历史记录将使Git和你的队友无法跟踪添加到该功能的任何后续提交。
其他开发人员的任何更改都需要合并git merge
而不是git rebase
。
因此,在提交拉取请求之前,通常使用交互式rebase清理代码是个好的办法。注意使用顺序。
集成已批准的功能
在你的团队批准某项feature后,你可以选择将该功能rebase到master
分支的顶端,然后git merge
再将该功能集成到主代码库中。
这与将上游更改合并到feature分支中的情况类似,但由于你不允许在master
分支中重写提交,因此你必须最终使用git merge
该功能进行集成。但是,通过在merge之前执行rebase,你可以确保会以fast-forward
方式merge,从而产生完美的线性历史记录。
fast-forward
指的是merge之后不会出现新的commit对象,分支的合并线是一条直线,被合并的分支,没有分叉,会丢失,再看不到该分支历史。相对地,no-fast-forward
方式merge之后会出现一个merge commit对象,分支的合并线在主干外出现一条分叉,在merge节点处合并为主干。
如果您不熟悉git rebase
,可以随时在临时分支中执行rebase。这样,如果你不小心弄乱了功能的历史记录,可以查看原始分支,然后重试。例如:
1 | git checkout feature |
总结
如果你更喜欢没有不必要的干净的合并提交,线性历史记录,你就需要开始了解使用rebase功能。同时你应该使用git rebase
而不是git merge
集成来自另一个分支的更改。
另一方面,如果你想保留项目的完整历史记录并避免重写公共提交的风险,你可以坚持使用git merge
。这两种选择都是完全有效的,但至少现在你可以选择利用git rebase
的好处 。
非常感谢你的阅读,辛苦了!
本文转自公众号「日拱一兵」的文章git rebase VS git merge?更优雅的git合并方式值得拥有