Git 合并到底使用Merge还是Rebase
liuian 2025-03-19 00:03 25 浏览
git rebase命令常常因为江湖上关于它是一种Git魔法命令的名声而导致Git新手对它敬而远之,但是事实上如果一个团队能够正确使用的话,它确实可以让生活变得更简单。在这篇文章中我们会比较git rebase和经常与之相提并论的git merge命令,并且在真实典型的Git工作流程中识别潜在的可使用rebase的场景。
概念概述
首先我们应该明白git rebase是用来处理git merge命令所处理的同样的问题。这两个命令都用于把一个分支的变更整合进另一个分支——只不过他们达成同样目的的方式不同。
请考虑这个场景,当你开始在一个专有的分支开发新的功能时,另一位团队成员更新了main分支的内容。这将会造成一个分叉的提交历史,对于任何一个使用Git作为代码协作工具的人来说都不会陌生。
现在假设main分支内新增的内容与你正在开发的新功能有关。为了把main分支里新增的代码应用在你的feature分支,你有两种方法:merge 和 rebase。
使用merge
最简单的方法就是把main分支合并进功能分支:
git checkout feature
git merge main或者用下面这样的单行命令:
git merge feature main这会在feature分支中创建一个合并提交,这次提交会连结两个分支的提交历史,在分支图示结构中看起来像下面这样:
合并操作很友好,因为它没有破坏性。现存的分支历史不会发生什么改变。这一特性避免了rebase操作的所有缺陷(下面会详细讨论)。
但是另一方面来说,这也意味着每当feature分支需要应用上游分支的更改时,都会在提交历史上增加一个无关的提交历史。如果main分支的更新非常活跃,这种操作也会对功能分支的提交历史产生相当程度的污染。虽然通过复杂的git log命令可以减轻这种提交历史的混乱现状,但仍然会让其他开发者对于提交历史感到费解。
使用rebase
为了替代merge操作,你也可以把feature分支的提交历史rebase到main分支的提交历史顶端:
git checkout feature
git rebase main这些操作会把feature分支的起始历史放到main分支的最后一次提交之上,也达成了使用main分支中新代码的目的。但是,相对于merge操作中新建一个合并提交,rebase操作会通过为原始分支的每次提交创建全新的提交,从而重写原始分支的提交历史。
使用rebase操作的最大好处在于你可以让项目提交历史变得非常干净整洁。首先,它消除了git merge操作所需创建的没有必要的合并提交。其次,正如上图所示,rebase会造就一个线性的项目提交历史——也就是说你可以从feature分支的顶部开始向下查找到分支的起始点,而不会碰到任何历史分叉。这在使用git log,git bisect以及gitk等命令时更简单。
不过为了获得这种便于理解的提交历史,却需要付出两种代价:安全性和可追溯性。如果不能遵循rebase的黄金法则,重写项目提交历史会为协作工作流程带来潜在的灾难性后果。再次,rebase操作丢失了合并提交能够提供的上下文信息——所以你就无法知道功能分支是什么时候应用了上游分支的变更。
可交互式rebase
可交互式rebase让你在把变更提交给其他分支之前有机会对提交记录进行修改。这甚至比自动rebase操作更强大,毕竟它提供了对于分支提交历史的完全掌控力。通常来说这一操作的使用场景在于合并功能分支到main分支之前,对于功能分支杂乱的提交记录进行整理。
进行可交互式rebase操作,需要向git rebase命令传递i选项参数
git checkout feature
git rebase -i main执行以上命令会打开一个文本编辑器,其中内容为分支中需要移动的所有提交列表:
pick 33d5b7a Message for commit #1
pick 9480b3d Message for commit #2
pick 5c67e61 Message for commit #3上面这样的列表正表示了分支被rebase之后其历史的长相。通过修改pick命令或者对提交历史进行重新排序,你可以让最终的提交历史变成任何你希望的样子。比如说,如果第二次提交修复了第一次提交的什么BUG,你可以使用fixup命令替代pick来把两次提交压缩在一起。
pick 33d5b7a Message for commit #1
fixup 9480b3d Message for commit #2
pick 5c67e61 Message for commit #3当你保存并关闭这个文件之后,Git会根据你的调改结果执行rebase操作,根据上面的例子项目历史会变成下图这样:
通过清除那些并不重要的提交历史可以让项目整体的历史更易读易懂。这一点是git merge操作所无法提供的。
rebase操作黄金法则
一旦你明白了什么是rebase,接下来最重要的事情就是要了解什么情况下不应该使用它。关于git rebase的黄金法则就是永远不要在公共分支上使用它。
举例来说,想一想如果把main分支rebase到feature分支之上,会发生什么:
rebase命令会把main分支中的所有提交都放到feature分支的提交记录顶端。问题在于这个改变目前只出现在你的本地仓库。其他开发者仍然在原来的main分支上进行开发。由于rebase会产生全新的提交记录,所以Git会认为现在你本地的main分支与所有其他人的产生了分叉。
唯一能够同步两个不同的main分支的方式就是将其合并起来,这会产生一个冗余的合并提交,并且这次合并中的大部分提交内容都是相同的(以前的main分支和你本地的main分支中)。不用说,这下可真让人疑惑。
所以任何时候要执行git rebase命令之前,先确认“是否有其他人也正在使用此分支?”如果答案是确定的,那么你就应该停下来想想有没有其他非破坏性的操作(比如试试git revert命令)。除了这样的情况之外,重写提交历史都是安全的。
Force-Pushing
如果你确实对main分支进行了rebase操作,然后想把main分支推送到远程仓库。这时Git会因为本地分支的提交与远程分支的提交发生了冲突,而阻止你这次的推送。但是,你仍然可以通过使用--force选项来强行进行推送,像这样:
# Be very careful with this command! git push --force强制推送的结果会让远程仓库的main分支使用被你rebase过的分支提交历史,当然这会让团队其他成员非常困惑。所以除非你明确知道你在做什么,否则不要轻易使用强制推送选项。
只有一种情况是属于“应当”使用强制推送命令的,那就是当你向远程仓库推送了一个私有分支之后,又做了一些清理工作。此时你大概的想法是:“哦!我发现在还是用现在这个分支的记录比较合适,不要之前已经推送的那个分支记录了”。即便如此,确定没有人与你在这个分支上进行协作仍然是非常重要的一件事情。
工作流实战
无论团队规模大小,rebase操作可以顺畅的接入现有团队的工作流程。在本部分中,我们一起看看在不同的功能开发阶段中,rebase都能提供哪些收益。
在任何一种工作流中,如果我们希望让rebase介入其中,那么第一步就是为功能开发创建一个专用分支。这样可以提供必要的分支结构以便安全地使用rebase:
本地清理
在现有工作流中包含rebase操作的最适合的场景之一是:清理本地正在进行中的开发分支。通过定期使用可交互rebase操作,可以清理本分支的提交记录,让每一次提交都更加聚焦并有意义。可交互rebase操作允许你在写代码的时候不用太在意提交历史,事实上你可以在事后再对提交历史进行清理。
当使用git rebase命令时,有两种选项可以作为新的base:功能分支的父分支(比如 main 分支),或者是本分支内历史中的某一次提交。第一种情况的示例我们在交互式rebase的段落见到过。后一种选项对于修改本分支内的提交历史则相当有用。比如下面的命令会开启一次对于最近三次提交历史的rebase操作。
git checkout feature git rebase -i HEAD~3通过指定HEAD~3作为rebase操作的新base,你并不是在实际移动分支——你只是以交互的方式对HEAD~3这次提交之后的三次提交历史进行重写。注意这个操作并不会将上游的修改引入feature分支:
如果你想对整个feature分支历史进行重写,那么应该试试git merge-base命令,它会返回给你feature分支的原始base。下面的命令返回原始base的commit ID,获得之后就可以用于git rebase命令的参数:
git merge-base feature main像上面这种rebase的使用场景非常利于将git rebase引入现有的工作流程,毕竟它只会影响本地分支。其他开发者能看到的只是你已经完成之后的作品,那种拥有干净提交历史,易于理解分支内容,便于跟踪开发过程的优美的分支提交历史。
不过仍然,只能对私有分支进行此操作。如果你通过同一分支与其他开发者进行协作,那么这个分支就是公共分支,是不允许重写提交历史的。
对于git merge操作没有可替代的方式用来清理本地的提交历史。
引入上游的修改
在本文最开始的部分,我们讨论过如何通过git merge或者git rebase方式引入上游main分支的修改。merge操作足够安全,因为它保留了完整的提交历史,但是rebase操作通过将功能分支的提交历史移到main分支的顶端从而创建了线性的提交历史。
此种对于git rebase操作的使用与清理本地提交历史类似(也可以同时操作),差别在于在执行过程中会引入上游main分支的提交。
请记住rebase可以对任何远端分支进行操作,并不仅限于main分支。比如当你需要与其他人协作开发一个功能时,你可以通过rebase来引入其他人的开发内容。
比如说,当你和另一个名叫John的开发者都对feature分支进行了提交动作,在你fetch远程的feature分支之后,本地仓库应该看起来是下图这样的:
为了整合这个分叉,你可以像对待main分支一样:要么通过merge操作将john/feature分支合并到本地feature分支,或者rebase本地feature分支到john/feature分支的顶端。
请注意这并不与rebase的黄金法则发生冲突,因为只有你本地的feature分支的新提交被移动到john/feature分支的顶端,新提交之前的所有提交历史都没有变化。这就好像说:“把我提交的新内容添加到John已经提交的内容之上。”在大多数情况下,这种操作比使用merge操作更符合人类的直觉。
git pull命令默认行为是进行一次合并操作,但你可以通过添加--rebase选项指定pull操作的行为为rebase。
使用pull request进行功能审查
如果你使用pull request来进行代码审查工作,那么在创建了pull request之后应该避免使用git rebase。一旦你创建了pull request,其他开发者就会来查看你的提交,也就意味着此时的分支算作是一个公共分支了。那么此时重写提交历史,则会让Git和团队成员无法判断哪些提交是属于这个功能的。
引入任何他人的修改时,应该使用git merge而不是git rebase。
因此在提交pull request之后进行一次交互式rebase来清理提交历史通常是一个好主意。
整合审查通过的功能
被团队审查通过的功能代码,可以先使用rebase将新代码移动到main分支的顶端,然后在进行git merge合并新功能到main分支中。
这个操作跟rebase上游分支到本地功能分支类似,只是由于你不能重写main分支的提交历史,所以你只能在最后通过git merge操作来把功能分支的代码整合进main分支。不过在合并之前进行一次rebase,可以保证这次merge操作是可以快速前进的,这样提交历史看上去就是完美的线性。这也给你机会可以在真正合并之前进行一次提交历史的清理。
如果你还不是很适应git rebase操作,那么总是可以利用一个临时分支来进行rebase操作。这样的话,万一你不小心搞乱了功能分支的提交历史,总还有兜底的机会从原始的功能分支再来一遍。就像下面这样:
git checkout feature
git checkout -b temporary-branch
git rebase -i main
# [Clean up the history]
git checkout main
git merge temporary-branch总结
这就是你开始使用rebase时所有需要了解的知识了。如果你希望一个干净线性的提交历史,而不是含有众多合并提交相互交织的提交历史,那么应该尝试在整合分支时使用git rebase而不是git merge。
反过来说,如果你想要保存完整的提交历史,避免重写公共提交的历史,仍然可以坚持使用git merge。两者都可以,但至少你现在拥有了另一个选项,可以见机利用 git rebase的优势。
相关推荐
- 搭建一个20人的办公网络(适用于20多人的小型办公网络环境)
-
楼主有5台机上网,则需要一个8口路由器,组网方法如下:设备:1、8口路由器一台,其中8口为LAN(局域网)端口,一个WAN(广域网)端口,价格100--400元2、网线N米,这个你自己会看了:)...
- 笔记本电脑各种参数介绍(笔记本电脑各项参数新手普及知识)
-
1、CPU:这个主要取决于频率和二级缓存,频率越高、二级缓存越大,速度越快,现在的CPU有三级缓存、四级缓存等,都影响相应速度。2、内存:内存的存取速度取决于接口、颗粒数量多少与储存大小,一般来说,内...
- 汉字上面带拼音输入法下载(字上面带拼音的输入法是哪个)
-
使用手机上的拼音输入法打成汉字的方法如下:1.打开手机上的拼音输入法,在输入框中输入汉字的拼音,例如“nihao”。2.根据输入法提示的候选词,选择正确的汉字。例如,如果输入“nihao”,输...
- xpsp3安装版系统下载(windowsxpsp3安装教程)
-
xpsp3纯净版在采用微软封装部署技术的基础上,结合作者的实际工作经验,融合了许多实用的功能。它通过一键分区、一键装系统、自动装驱动、一键设定分辨率,一键填IP,一键Ghost备份(恢复)等一系列...
- 没有备份的手机数据怎么恢复
-
手机没有备份恢复数据方法如下1、使用数据线将手机与电脑连接好,在“我的电脑”中可以看到手机的盘符。 2、将手机开启USB调试模式。在手机设置中找到开发者选项,然后点击“开启USB调试模式”。 3、...
- 电脑怎么激活windows11专业版
-
win11专业版激活方法有多种,以下提供两种常用的激活方式:方法一:使用激活密钥激活。在win11桌面上右键点击“此电脑”,选择“属性”选项。进入属性页面后,点击“更改产品密钥或升级windows”。...
- 华为手机助手下载官网(华为手机助手app下载专区)
-
华为手机助手策略调整,已不支持从应用市场下载手机助手,目前华为手机助手是需要在电脑上下载或更新手机助手到最新版本,https://consumer.huawei.com/cn/support/his...
- 光纤线断了怎么接(宽带光纤线断了怎么接)
-
宽带光纤线断了可以重接,具体操作方法如下:1、光纤连接的时候要根据束管内,同色相连,同芯相连,按顺序进行连接,由大到小。一般有三种连接方法,分别是熔接、活动连接和机械连接。2、连接的时候要开剥光缆,抛...
- win7旗舰版和专业版区别(win7旗舰版跟专业版)
-
1、功能区别:Win7旗舰版比专业版多了三个功能,分别是Bitlocker、BitlockerToGo和多语言界面; 2、用途区别:旗舰版的功能是所有版本中最全最强大的,占用的系统资源,...
- 万能连接钥匙(万能wifi连接钥匙下载)
-
1、首先打开wifi万能钥匙软件,若手机没有开启WLAN,就根据软件提示打开WLAN开关;2、打开WLAN开关后,会显示附近的WiFi,如果知道密码,可点击相应WiFi后点击‘输入密码’连接;3、若不...
- 雨林木风音乐叫什么(雨林木风是啥)
-
雨林木风的创始人是陈年鑫先生。陈年鑫先生于1999年创立了雨林木风公司,其初衷是为满足中国市场对高品质、高性能电脑的需求。在陈年鑫先生的领导下,雨林木风以技术创新、产品质量和客户服务为核心价值,不断推...
- aics6序列号永久序列号(aics6破解序列号)
-
关于AICS6这个版本,虽然是比较久远的版本,但是在功能上也是十分全面和强大的,作为一名平面设计师的话,AICS6的现有的功能已经能够应付几乎所有的设计工作了……到底AICC2019的功能是不是...
- 手机可以装电脑系统吗(手机可以装电脑系统吗怎么装)
-
答题公式1:手机可以通过数据线或无线连接的方式给电脑装系统。手机安装系统需要一定的技巧和软件支持,一般需要通过数据线或无线连接的方式与电脑连接,并下载相应的软件和系统文件进行安装。对于大部分手机用户来...
- 一周热门
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
