美文网首页
Git - 回滚

Git - 回滚

作者: 齐晋 | 来源:发表于2018-09-29 18:40 被阅读41次

    回滚命令

    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 - 回滚

          本文链接:https://www.haomeiwen.com/subject/kppboftx.html