回滚命令
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
一致
场景
当修改了一些内容,并且提交到了本地。但是后来发现这些修改不用了,想完全回滚掉。
解决方法
这时,就可以通过如下命令,使本地分支的HEAD
与远程分支的HEAD
保持一致。
git reset --hard origin/master
命令执行完成后,本地分支的代码就与远程分支完全一样了。
回滚Merge
回滚最新的Merge
场景
假设有如下流程:
两个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
场景
- 在test1分支修改了一些东西
- 将test1分支merge到master
- 又有很多其他分支被merge到了master
这时master分支的git log如下图所示:
现在发现test1分支有重大bug,需要回滚。而merge完test1后,又有很多分支被merge到了master中。全量回滚不现实。那么怎么只回滚test1的提交呢?
解决方法
每个Merge操作会产生一个Merge commit。 Merge commit都会有Merge
这个单词在git log中。如下图所示:
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
会有如下结果:
这样,这个merge就被回滚了。
注:当有conflict的时候会比较麻烦