Git - 回滚

回滚命令

git回滚的命令主要有如下几种:

  • git reset
  • git revert

git revert常用于回滚单个commit。详情见 Git - revert
git reset常用于回滚多个commit,或者回滚后重新修改。详情请见 Git - reset

场景

回滚某个commit

场景
提交了很多commit,想把某次commit的代码回滚。

commit 8abfadc33abe671ac44ca3c71e6310aac067c75d
commit ec3efbc7ba89a537c8f8d8a0f92102f36d5a1278
commit 66a69a7369dc4bd49ce44c83bb1bdc19c4600659

比如想回滚ec3efbc7ba89a537c8f8d8a0f92102f36d5a1278这个commit。

解决方法
可以用git revert命令

git revert ec3efbc7ba89a537c8f8d8a0f92102f36d5a1278

git revert会创建一个新的commit,这个新commit与原来commit操作完全相反,达到回滚效果。下面是revert时创建的新的commit:

commit e16a5425c124ae7af713ffbc2e4faf2ec33bb25d (HEAD -> test1)
Author: micheal <michealyang@qijin.com>
Date:   Sat Sep 29 17:25:57 2018 +0800

    Revert "[test1] 6"

    This reverts commit ec3efbc7ba89a537c8f8d8a0f92102f36d5a1278.

回滚到某个commit

场景
提交了很多commit,想把最近几次的commit都回滚掉。

commit 8abfadc33abe671ac44ca3c71e6310aac067c75d
commit ec3efbc7ba89a537c8f8d8a0f92102f36d5a1278
commit f50abb74fbf44c5d30a8e2fc3a42ebc3867f8a38

如,将代码回滚到f50abb74fbf44c5d30a8e2fc3a42ebc3867f8a38这次提交。

解决方法
有两种思路:

  • 需要回滚几个commit
  • 需要回滚到哪个commit

所谓“需要回滚几个commit”,是指知道哪几个commit是需要回滚的。有了这个数量,可以用如下命令回滚:

git reset HEAD~2

git reset HEAD@{2}

表示要回滚2个commit。这样正好回滚到f50abb74fbf44c5d30a8e2fc3a42ebc3867f8a38

所谓“需要回滚到哪个commit”,是指不需要知道要回滚几个commit,只需要知道要回滚到的commit id即可。上面的例子要回滚到需要回滚到f50abb74fbf44c5d30a8e2fc3a42ebc3867f8a38这个commit。那么,如下命令可以做到:

git reset f50abb74fbf44c5d30a8e2fc3a42ebc3867f8a38

这两种思路操作不同,但是最终达到的效果是一样的。

当然,根据需要,可以添加--hard, --soft--mixed等参数。
如:

git reset --mixed HEAD~2
等同于
git reset HEAD~2

表示回滚掉的代码依旧保留,还可以修改

git reset --hard HEAD~2

表示抛弃回滚掉的代码。

reset完成后,就可以通过git push把回滚后的结果同步到远程分支了。

恢复到与HEAD一致

场景
当修改了一些内容,并且提交到了本地。但是后来发现这些修改不用了,想完全回滚掉。

image.png

解决方法
这时,就可以通过如下命令,使本地分支的HEAD与远程分支的HEAD保持一致。

git reset --hard origin/master

命令执行完成后,本地分支的代码就与远程分支完全一样了。

回滚Merge

回滚最新的Merge

场景
假设有如下流程:

image.png

两个branch test2和test1分别被merge到了master中,test1最后merge
现在需要把test1的merge回滚掉。

解决方法
可使用如下任何一种方法:

git reset --merge ORIG_HEAD
git reset --hard ORIG_HEAD
git reset --hard HEAD~1
git reset --hard HEAD@{1}

上面三种命令都能实现,不过还是需要讲解下参数:

  • HEAD是当前最新的commit。ORIG_HEAD是上一次的commit。因此revert ORIG_HEAD的意思就是回滚到上次commit。ORIG_HEAD现在已经不常用了,完全可以用更灵活的HEAD@{1}替代
  • HEAD~n表示倒数第几个commit。HEAD~1就表示上一次commit,HEAD~2表示上上次commit,以此类推。HEAD@{n}与之同理

回滚中间的Merge

场景

  1. 在test1分支修改了一些东西
  2. 将test1分支merge到master
  3. 又有很多其他分支被merge到了master

这时master分支的git log如下图所示:


image.png

现在发现test1分支有重大bug,需要回滚。而merge完test1后,又有很多分支被merge到了master中。全量回滚不现实。那么怎么只回滚test1的提交呢?

解决方法
每个Merge操作会产生一个Merge commit。 Merge commit都会有Merge这个单词在git log中。如下图所示:

image.png

Merge单词后面还有两个commit号,是本次commit的parent commit。
所谓parent commit是指,master分支merge了test1分支,那么master分支中的最新commit和test1分支中的最新commit就是生成的Merge commit的parent commit。

上图中所示的7c544ac是master分支的最新commit。7b68089是test1分支的最新commit。在这里我们可以单独把其中的一个parent给revert掉。序号从1开始。

git revert -m 2 346774d0ccc018e4b0fb40695587cbd82d4299b3

其中-m参数后面跟着的就是要revert的parent commit的序号。
我们知道,revert命令会创建一个新的commit,执行完上面的操作后,查看git log会有如下结果:

image.png

这样,这个merge就被回滚了。

注:当有conflict的时候会比较麻烦

参考

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 方法一,删除远程分支再提交 ①首先两步保证当前工作区是干净的,并且和远程分支代码一致 $ git co curre...
    那小白阅读 1,794评论 0 0
  • git常用命令 GIT常用命令备忘:http://stormzhang.com/git/2014/01/27/gi...
    新篇章阅读 12,761评论 1 26
  • git作为版本控制肯定是常见的了,对于git有一个版本回滚的功能是我们在开发过程中应该用到比较多,掌握它对于我们开...
    草帽小子J阅读 4,396评论 0 3
  • 学明行日记第 40篇 恩恩读经105周第5天 读经日期:2018年9月21日 星期五 阴 亲子读经内容: 1.易经...
    点亮喜乐富足的人生阅读 1,371评论 0 0
  • 在这无比美妙的端午佳节之日,外面无比炎热,在还算凉爽的宿舍里打开电脑看了朋友推荐的一部电影《逆光飞翔》。电影讲述了...
    Twinkle_______阅读 3,392评论 0 0