前言
git reset
和git revert
都是用来撤销某一次的提交操作,但是他们撤销的方法有所不同。
1.git revert
git revert
的撤销操作是通过一个新的提交来完成的,这个提交中的改动和要撤销的改动是相反的,以此实现指定提交的撤销。
举例:
从图片中可以看到我的操作是在代码中依次填入两行改动并进行了两次本地的提交(蓝色是本地仓库我的分支,紫色是远程仓库我的分支)。
接下来我会执行 git revert <指定操作的commit ID>:
arduino
git revert 8700d52fbda866a1c481eca607b2f2a759297a8c //改动二
按下 Esc 键
输入 :wq
然后按 Enter
从图片中可以看出
git revert
的操作就是向本地仓库生成一个新的Revert提交,其内容就是需要撤销的提交的反操作。
后三行的操作对应的是下图这个区域:
特点:
- 不会修改历史: 由于
git revert
是通过一个新提交来撤销之前的提交,所以它会保留住之前的提交历史。 - 适合公共分支: 如果你和他人都在一个公共分支上协作,由于
git revert
会保留提交历史,每次改动记录都存在,是十分安全的操作。 - 其他操作:
git revert --continue
:冲突发生之后,解决冲突之后git add
解决后的文件(提交到暂存区),然后输入git revert --continue
继续操作(前文我revert 改动一 就会发生冲突);git revert --abort
:冲突发生之后,不想继续这次的revert操作;git revert -n <改动一的commit ID>^..<改动三的commit ID>
:撤销改动一到改动三(包含)之间的所有提交,它会将所有反向更改暂时保存在暂存区中,但不会立即提交。你可以手动编辑这些更改,并最终一起提交。
2.git reset
git reset
的撤销操作是通过回退本地分支指针实现,回退到指定的一次提交之后,之前的提交操作会被丢弃。
举例:
从图片中可以看到我的操作是在代码中依次填入三行改动并进行了三次本地的提交(蓝色是本地仓库我的分支,紫色是远程仓库我的分支)。
接下来我会执行 git reset --hard <指定操作的commit ID> : 改动二的commit ID:7b1b296e83530bd016c5787d2de7da77ac48ee79
css
git reset --hard 7b1b296e83530bd016c5787d2de7da77ac48ee79
从图片中可以看出我们本地仓库的提交中改动三没有了,说明上述
git reset --hard
的操作是实现的效果是分支本地指针指向选定的提交,之后的其他提交都丢弃。
终端返回的内容HEAD is now at 7b1b296e8 🐛 change:改动二
。
特点:
- 会修改历史: 由于
git reset
会丢弃一些后面的提交,所以这个操作是会改变提交历史。 - 适用于本地操作: 由于
git reset
改动大,会修改历史,所以更适合回滚自己本地的提交,不适合对已经推送到远程仓库的提交进行改动。 - 其它操作:
HEAD~X
: 撤销最近X次提交
perl
git reset HEAD~2
可以看到把改动一和改动二的提交给撤销了,但是在改动区保留了这两次提交的改动。
git reset --soft <commit ID>
: 只移动分支指针,暂存区和工作目录中的更改都会保留,以便再次提交。
scss
git reset --soft 8700d52fbda866a1c481eca607b2f2a759297a8c //改动二的ID
可以看到左边将改动放到了暂存区,右边工作区修改同样保留了改动三。
git reset --mixed(或不带选项默认) <commit ID>
:移动分支指针,不会修改工作区,撤销了提交和暂存的更改,但保留了工作区的修改。
可以看到左边将改动放在改动区,并没有放入暂存区,右边工作区是保留了改动三。
附录:
在已经提交到远程仓库的情况下,使用git reset
之后,只有本地指针会回退,如果想让远端仓库也同样回退可以使用git push origin HEAD --force
:强制推送到远程仓库覆盖远程历史(慎用!)。