Git - 回滚

  • 回滚命令
  • 场景
    • 回滚某个commit
    • 回滚到某个commit
    • 恢复到与HEAD一致
    • 回滚Merge
      • 回滚最新的Merge
      • 回滚中间的Merge
  • 参考

回滚命令

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 
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一致

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

Git - 回滚_第1张图片
image.png

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

git reset --hard origin/master

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

回滚Merge

回滚最新的Merge

场景
假设有如下流程:

Git - 回滚_第2张图片
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如下图所示:


Git - 回滚_第3张图片
image.png

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

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

Git - 回滚_第4张图片
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会有如下结果:

Git - 回滚_第5张图片
image.png

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

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

参考

  • Git Document - git reset
  • Undo a Git merge that hasn't been pushed yet
  • How to revert a merge commit that's already pushed to remote branch?
  • 版本回退

你可能感兴趣的:(Git - 回滚)