1 简介
最近在使用Git时遇到了远程分支需要版本回滚的情况,于是做了一下研究,写下这篇博客。
2 问题
如果提交了一个错误的版本,怎么回退版本?
如果提交了一个错误的版本到远程分支,怎么回退远程分支版本?
如果提交了一个错误的版本到公共远程分支,又该怎么回退版本?
3 本地分支版本回退的方法
如果你在本地做了错误提交,那么回退版本的方法很简单
先用下面命令找到要回退的版本的commit id:
<span style="color:#000000"><code class="language-git">git reflog </code></span>
接着回退版本:
<span style="color:#000000"><code class="language-git"><span style="color:#880000">git</span> <span style="color:#880000">reset</span> <span style="color:#006666">-</span><span style="color:#006666">-</span><span style="color:#880000">hard</span> <span style="color:#880000">ad0ef3e</span></code></span>
ad0ef3e就是你要回退的版本的commit id的前面几位
如果直接在eclipse上操作呢,如使用的egit插件,又该怎么做?
很多人看到这里有三个选项,不禁有个疑问,都有什么讲究吗,答案是肯定的。
Soft - 当前分支重置到指定commit记录位置,索引和工作树不变;
Mixed - 当前分支重置到指定commit记录位置,索引被更新,工作树不变;
Hard - 当前分支重置到指定commit记录位置,索引和工作树都更新。
貌似不好理解,首先要理解GIT的三个区域(工作树、索引区、仓库),可参考廖雪峰的《Git教程》
先做soft的测试,新建Soft.java文件,可以看到此文件未添加到索引控制
先进行一次提交,提交后在History窗口中重置此次提交,选择soft:
重置后查看工作树,如图
从上图可以看出,soft文件还存在,说明重置没有改变工作树,而且soft文件不是“问号”图标,说明已经添加到索引,说明索引也没有变。唯一重置的是历史记录。
然后新建Mixed.java文件,此时Mixed.java也没有添加到索引控制,然后提交。
在History窗口中重置
重置后查看工作树结果如下:
从上图可以看出,Mixed.java文件还存在,说明工作树没有改变,但是文件状态是untracked,说明索引被更新,此时文件没有添加索引控制。
最后来看hard重置,新建Hard.java文件,此时文件没有添加索引,然后提交。
在History界面重置此次提交,如图:
重置后再查看工作树,结果如下:
可以看到Hard.java文件已经不存在了,说明索引和工作树都被更新。
4 自己的远程分支版本回退的方法
如果你的错误提交已经推送到自己的远程分支了,那么就需要回滚远程分支了。
首先要回退本地分支:
<span style="color:#000000"><code class="language-git"><span style="color:#880000">git</span> <span style="color:#880000">reflog</span>
<span style="color:#880000">git</span> <span style="color:#880000">reset</span> <span style="color:#006666">-</span><span style="color:#006666">-</span><span style="color:#880000">hard</span> <span style="color:#880000">ad0ef3e</span></code></span>
紧接着强制推送到远程分支:
<span style="color:#000000"><code class="language-git">git push -f</code></span>
注意:本地分支回滚后,版本将落后远程分支,必须使用强制推送覆盖远程分支,否则无法推送到远程分支
那如何使用eclipse插件egit操作呢?
前面的回退操作上面已经讲到,那我们可以直接推送到远程仓库吗?不妨一试。结果如下图。
看来只能强制推送了,做法如下
现在可以可以看到已经推送成功了
5 公共远程分支版本回退的问题
看到这里,相信你已经能够回滚远程分支的版本了,那么你也许会问了,回滚公共远程分支和回滚自己的远程分支有区别吗?
答案是,当然有区别啦。
一个显而易见的问题:如果你回退公共远程分支,把别人的提交给丢掉了怎么办?
下面来分析:
假如你的远程master分支情况是这样的:
A1–A2–B1
其中A、B分别代表两个人,A1、A2、B1代表各自的提交。并且所有人的本地分支都已经更新到最新版本,和远程分支一致。
这个时候你发现A2这次提交有错误,你用reset回滚远程分支master到A1,那么理想状态是你的队友一拉代码git pull,他们的master分支也回滚了,然而现实却是,你的队友会看到下面的提示:
<span style="color:#000000"><code class="language-git">$ git status
On branch master
Your branch is ahead of <span style="color:#009900">'origin/master'</span> <span style="color:#000088">by</span> <span style="color:#006666">2</span> commits.
(use <span style="color:#009900">"git push"</span> <span style="color:#4f4f4f">to</span> publish your <span style="color:#4f4f4f">local</span> commits)
nothing <span style="color:#4f4f4f">to</span> commit, working <span style="color:#4f4f4f">directory</span> clean</code></span>
(1) 这个时候,你大吼一声:兄弟们,老子回退版本了。如果你的队友都是神之队友,比如: Tony(腾讯CTO),那么Tony会冷静的使用下面的命令来找出你回退版本后覆盖掉的他的提交,也就是B1那次提交:
<span style="color:#000000"><code class="language-git">git reflog</code></span>
然后冷静的把自己的分支回退到那次提交,并且拉个分支:
<span style="color:#000000"><code class="language-git">git checkout tony_branch <span style="color:#880000">//先回到自己的分支 </span>
git reflog <span style="color:#880000">//接着看看当前的commit id,例如:0bbbbb </span>
git reset <span style="color:#4f4f4f">--</span>hard B1 <span style="color:#880000">//回到被覆盖的那次提交B1</span>
git checkout -b tony_backup <span style="color:#880000">//拉个分支,用于保存之前因为回退版本被覆盖掉的提交B1</span>
git checkout tony_branch <span style="color:#880000">//拉完分支,迅速回到自己分支</span>
git reset <span style="color:#4f4f4f">--</span>hard <span style="color:#006666">0</span>bbbbbb <span style="color:#880000">//马上回到自己分支的最前端</span></code></span>
通过上面一通敲,Tony暂时舒了一口气,还好,B1那次提交找回来了,这时tony_backup分支最新的一次提交就是B1,接着Tony要把自己的本地master分支和远程master分支保持一致:
<span style="color:#000000"><code class="language-git"><span style="color:#880000">git</span> <span style="color:#880000">reset</span> <span style="color:#006666">-</span><span style="color:#006666">-</span><span style="color:#880000">hard</span> <span style="color:#880000">origin/master</span></code></span>
执行了上面这条命令后,Tony的master分支才真正的回滚了,也就是说你的回滚操作才能对Tony生效,这个时候Tony的本地maser是这样的:
A1
接着Tony要再次合并那个被丢掉的B1提交:
<span style="color:#000000"><code class="language-git">git checkout master <span style="color:#880000"> //切换到master</span>
git <span style="color:#4f4f4f">merge</span> tony_backup <span style="color:#880000"> //再合并一次带有B1的分支到master</span></code></span>
ony终于长舒一口气,这个时候他的master分支是下面这样的:
A1 – B1
终于把丢掉的B1给找回来了,接着他push一下,你一拉也能同步。
同理对于所有队友也要这样做,但是如果该队友没有提交被你丢掉,那么他拉完代码git pull之后,只需要强制用远程master覆盖掉本地master就可以了:
<span style="color:#000000"><code class="language-git"><span style="color:#880000">git</span> <span style="color:#880000">reset</span> <span style="color:#006666">-</span><span style="color:#006666">-</span><span style="color:#880000">hard</span> <span style="color:#880000">origin/master</span></code></span>
(2) 然而很不幸的是,现实中,我们经常遇到的都是猪一样的队友,他们一看到下面提示:
<span style="color:#000000"><code class="language-git">$ git status
On branch master
Your branch is ahead of <span style="color:#009900">'origin/master'</span> <span style="color:#000088">by</span> <span style="color:#006666">2</span> commits.
(use <span style="color:#009900">"git push"</span> <span style="color:#4f4f4f">to</span> publish your <span style="color:#4f4f4f">local</span> commits)
nothing <span style="color:#4f4f4f">to</span> commit, working <span style="color:#4f4f4f">directory</span> clean</code></span>
就习惯性的git push一下,或者他们直接用的SourceTree这样的图形界面工具,一看到界面上显示的是推送的提示就直接点了推送按钮,卧&槽,你辛辛苦苦回滚的版本就这样轻松的被你猪一样的队友给还原了,所以,只要有一个队友push之后,远程master又变成了:
A1 – A2 – B1
这就是分布式,每个人都有副本。这个时候你连揍他的心都有了,怎么办呢?你不能指望每个人队友都是git高手,下面我们用另外一种方法来回退版本。
注意:博主是在虚拟机中实验的,用于模拟两个人的操作,如果你在一个机器上,用同一个账号在不同的目录下克隆两份代码来实验的话,回退远程分支后,另外一个人是不会看到落后远程分支两次提交的,所以请务必使用虚拟机来模拟A、B两个人的操作
6 公共远程分支版本回退的方法
使用git reset回退公共远程分支的版本后,需要其他所有人手动用远程master分支覆盖本地master分支,显然,这不是优雅的回退方法,下面我们使用另个一个命令来回退版本:
<span style="color:#000000"><code class="language-git">git revert HEAD <span style="color:#880000">//撤销最近一次提交</span>
git revert HEAD~<span style="color:#006666">1</span> <span style="color:#880000">//撤销上上次的提交,注意:数字从0开始</span>
git revert <span style="color:#006666">0f</span>faacc <span style="color:#880000">//撤销0ffaacc这次提交</span></code></span>
git revert 命令意思是撤销某次提交。它会产生一个新的提交,虽然代码回退了,但是版本依然是向前的,所以,当你用revert回退之后,所有人pull之后,他们的代码也自动的回退了。
但是,要注意以下几点:
- revert 是撤销一次提交,所以后面的commit id是你需要回滚到的版本的前一次提交
- 使用revert HEAD是撤销最近的一次提交,如果你最近一次提交是用revert命令产生的,那么你再执行一次,就相当于撤销了上次的撤销操作,换句话说,你连续执行两次revert HEAD命令,就跟没执行是一样的
- 使用revert HEAD~1 表示撤销最近2次提交,这个数字是从0开始的,如果你之前撤销过产生了commi id,那么也会计算在内的。
- 如果使用 revert 撤销的不是最近一次提交,那么一定会有代码冲突,需要你合并代码,合并代码只需要把当前的代码全部去掉,保留之前版本的代码就可以了.
git revert 命令的好处就是不会丢掉别人的提交,即使你撤销后覆盖了别人的提交,他更新代码后,可以在本地用 reset 向前回滚,找到自己的代码,然后拉一下分支,再回来合并上去就可以找回被你覆盖的提交了。
7 revert 合并代码,解决冲突
使用revert命令,如果不是撤销的最近一次提交,那么一定会有冲突,如下所示:
<span style="color:#000000"><code class="language-git"><<<<<<< HEAD
全部清空
<span style="color:#009900">第一次提交
=======</span>
全部清空
>>>>>>> parent of c24cde7... 全部清空</code></span>
解决冲突很简单,因为我们只想回到某次提交,因此需要把当前最新的代码去掉即可,也就是HEAD标记的代码:
<span style="color:#000000"><code class="language-git"><<<<<<< HEAD
全部清空
<span style="color:#009900">第一次提交
=======</span></code></span>
把上面部分代码去掉就可以了,然后再提交一次代码就可以解决冲突了。
8 继续扩展,简单粗暴的回滚方法
看到这里也许你已经觉得学会了远程仓库版本回滚方法了,但是实践中总是会遇到很多不按套路来的问题,考虑下面一种情况:
如果你们开发中,忽然发现前面很远的地方有一次错误的合并代码,把本来下一次才能发的功能的代码合并到了这一次来了,这个时候全体成员都觉得直接回滚比较快,因为他们都有备份,覆盖了无所谓,这个时候用reset的话对队友的要求比较高,用revert的话呢要大面积的解决冲突,也很麻烦呀,怎么办呢?
这个时候,可以使用简单粗暴的办法,直接从那个错误的提交的前一次拉取一份代码放到其他目录,然后将master代码全部删除,把那份新代码方进去,然后提交,果然简单粗暴啊,虽然这种方法不入流,但是,实践中发现很好使啊,所以,实践是检验真理的唯一标准。遇到问题还是要灵活应对。
9 总结
远程分支回滚的三种方法:
- 自己的分支回滚直接用reset
- 公共分支回滚用revert
- 错的太远了直接将代码全部删掉,用正确代码替代