Git使用Merge和Rebase区别及心得技巧

news2025/1/13 8:00:47

git rebase命令常常因为江湖上关于它是一种Git魔法命令的名声而导致Git新手对它敬而远之,但是事实上如果一个团队能够正确使用的话,它确实可以让生活变得更简单。在这篇文章中我们会比较git rebase和经常与之相提并论的git merge命令,并且在真实典型的Git工作流程中识别潜在的可使用rebase的场景。

Merge和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分支的更新非常活跃,merge会对功能分支的提交历史产生相当程度的污染(人多并行开发分支多时)。虽然通过复杂的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的优势。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/176863.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【回望2022,走向2023】一个双非二本非科班的学生的旅途

目录 1.自我介绍 2.高考与暑假 梦想 幻灭 决心 暑假 3.大一上学期 4.奋进之路 5.展望未来 1.自我介绍 我是一个双非本科的大一学生,在2023年的新春之际,借着CSDN的这次年度总结活动,来好好回顾一下,2022这个平凡却又不乏…

css 2D转换

文章目录一、什么是2D转换二、rotate() 方法(旋转)三、translate() 方法(位移)四、scale() 方法(缩放)五、skew() 方法 (倾斜)一、什么是2D转换 在二维空间下对元素进行移动、缩放、…

面试官问我有没有分布式系统开发经验,我一脸懵圈…

目录 从单块系统说起团队越来越大,业务越来越复杂分布式出现:庞大系统分而治之分布式系统所带来的技术问题一句话总结:什么是分布式系统设计和开发经验补充说明:中间件系统及大数据系统 前言 现在有很多Java技术方向的同学在找工…

深度学习网络---YOLO系列

深度学习网络—YOLO yolov1(仅适用一个卷积神经网络端到端地实现检测物体的目的) 首先将输入图片resize到448448,然后送入CNN网络,最后处理预测的结果得到检测的目标;yolov1的具体思想是将全图划分为SS的格子&#xf…

结构型模式-外观模式

1.概述 有些人可能炒过股票,但其实大部分人都不太懂,这种没有足够了解证券知识的情况下做股票是很容易亏钱的,刚开始炒股肯定都会想,如果有个懂行的帮帮手就好,其实基金就是个好帮手,支付宝里就有许多的基…

智能的本质不是数据算法算力和知识

编者按:人机之间未解决的大部分问题不是统计问题,而是统计概率分布外的问题。人是自然的,又不是自然的,还是社会的,人类和机器都可以作为认知的载体,但认知的性质是不同的,一个是生命的认知&…

GA6-BGSM/GPRS模块介绍

GA6-BGSM/GPRS模块简介GA6-B是一个4频的GSM/GPRS模块,工作的频段为:EGSM 900MHz、 GSM850MHz和DCS1800, PCS1900。GA6-B支持GPRS multi-slot class 10/ class 8(可选)和 GPRS 编码格式CS-1, CS-2, CS-3 and CS-4。模块的尺寸只有2…

SelectPdf for .NET 22.0 Crack

SelectPdf for .NET 是一个专业的 PDF 库,可用于创建、编写、编辑、处理和读取 PDF 文件,而无需在 .NET 应用程序中使用任何外部依赖项。使用此 .NET PDF 库,您可以实现丰富的功能,从头开始创建 PDF 文件或完全通过 C#/VB.NET 处理…

python数据结构——栈、队列

python数据结构——栈、队列、树和算法栈栈的操作队列单端队列操作双端队列操作链表或者顺序表的使用场景: 当数据需要后进先出,来构建栈或者先进先出,构建队列时 栈或者队列之内的数据可以以顺序表或者链表的方式进行存储 python内置的数据…

Python基础学习 -- 模块与包

1、模块每一个py文件都可以理解为一个模块,模块可以增加项目的可读性2、新建一个名为算数.py文件,代码内容如下:print("算数模块被加载!") def 加法(a,b):print(ab)3、新建一个main.py文件,调用模块的内容第…

Vue TypeScript 使用eval函数的坑

正常情况下,项目里不会用eval函数,但是万一要调用一个全局的js库,就需要用eval做些骚操作,这个时候编译会提示: is strongly discouraged as it poses security risks and may cause issues with minification. 警告是…

Java多线程(二)——ReentrantLock源码解析(补充1——从AQS中唤醒的线程)

ReentrantLock源码解析(补充1) 上一章仅介绍了 ReentrantLock 的常用方法以及公平锁、非公平锁的实现。这里对上一章做一些补充。主要是: AQS 中阻塞的线程被唤醒后的执行流程 (本篇讲述) 可打断的锁 lock.lockInter…

【QT5.9】与MFC对比学习笔记-感悟篇2【2023.01.23】

是对QT的分析,不仅局限于QT。 二者区别 天下文章一大抄,技术也一样。MFC是对Windows系统API进行的封装,是以视类与文档类为核心的框架设计。微软20年前就已经把MVC玩的很6了,还有控件、动态库等等技术都是微软爸爸先搞出来的。若…

Kubernetes:认识 K8s开源 Web/桌面 客户端工具 Headlamp

写在前面 分享一个 k8s 客户端开源项目 Headlamp 给小伙伴博文内容涉及: Headlamp 桌面/集群 Web 端安装启动导入集群简单查看集群信息 理解不足小伙伴帮忙指正 我所渴求的,無非是將心中脫穎語出的本性付諸生活,為何竟如此艱難呢 ------赫尔曼…

第八层:模板

文章目录前情回顾模板模板的概念模板的特点模板分类函数模板作用语法函数模板的使用注意事项普通函数和函数模板的区别普通函数和函数模板的调用规则优先调用普通函数空模板强调函数模板函数模板可以发生重载函数模板产生更好的匹配时模板的局限性类模板作用语法类模板实例化对…

Redis在秒杀场景的作用

秒杀业务特点:限时限量,业务系统要处理瞬时高并发请求,Redis是必需品。 秒杀可分成秒杀前、秒杀中和秒杀后三阶段,每个阶段的请求处理需求不同,Redis具体在秒杀场景的哪个环节起到作用呢? 1 秒杀负载特征…

Java-数据结构-二叉树<三>

承接上文: Java-数据结构-二叉树<一> Java-数据结构-二叉树<二> 一. 二叉树的简单介绍 见Java-数据结构-二叉树<一> 二. 二叉树的典型代码实现 见Java-数据结构-二叉树<一&#x…

4. RNN网络架构解读|词向量模型|模型整体框架|训练数据构建|CBOW和Skip-gram模型|负采样方案

文章目录RNN网络架构解读词向量模型模型整体框架训练数据构建CBOW和Skip-gram模型负采样方案RNN网络架构解读 递归神经网络实际上就是普通的神经网络的部分进行修改更新:实际上常用于时间序列的更新。或者就是自然处理中 X序列代表着时间序列,x0是一个时…

linux入门---云服务器购买和登陆

目录标题云服务器选择云服务器购买xshell下载如何登陆云服务器Linux的新建与删除新建删除云服务器选择 学习linux的时候云服务器是一个非常重要的工具,那么我们在购买云服务器的时候有很多选择比如说:华为云,腾讯云,阿里云等等&a…

【实操案例十二】类和对象 实例代码及运行效果图!

任务一:定义一个圆的类,计算面积和周长 # 任务一:定义一个圆的类,计算面积和周长 import math class Circle():def __init__(self,r):self.rrdef get_area(self):return math.pi*r*rdef get_perimeter(self):return 2*math.pi*r …