回滚是 Git 的杀手锏,是程序员的后悔药。那如何进行回滚呢?答案是 git revert
。接下来进行一步步演示:
- 通过
git init
创建仓库 - 然后
vi a.txt
创建一个文件,第一行写上a
,提交gc -am 'feat: first commit'
- 继续
vi a.txt
创建一个文件,第二行写上b
,提交gc -am 'fix: add b'
- 继续
vi a.txt
创建一个文件,第二行写上c
,提交gc -am 'fix: add c'
- 继续
vi a.txt
创建一个文件,第二行写上d
,提交gc -am 'fix: add d'
好了,此时 a.txt 的内容如下:
a
b
c
d
用 git log
查看历史提交记录:
commit 3ced7b954ec5023b827ba685a96f836bf1acefde (HEAD -> two, one)
fix: add d
commit 873c8bdb57149fabbd533caa7c4b6ec9ac3adb6a
fix: add c
commit 93ea0f972d591d54b07edba40a51b28486bbf9fd
fix: add b
commit c58aa1bbb6d22f6ba19b89252a352abfea8d92c4
feat: first commit
到目前为止,你的提交长这样:
是不是还挺好看的呢!我们把 commit 链描述如下:
A -> B -> C -> D
假如现在你想把最后一次提交 D 给回滚掉,应该如何操作呢?很简单:
$ git revert HEAD
这个时候,会自动生成一次提交,默认内容为:
Revert "fix: add d"
This reverts commit 3ced7b954ec5023b827ba685a96f836bf1acefde.
当然,这个是可以改的,我们先保持不动,用 :wq
保存并退出。会发现多了一次新的提交,该提交的作用就是把最后一次的提交给撤销了。
那如果不是撤销最后一次提交呢,中间任何一次都可以的,假如我们要回滚 add b 那次提交,可以用:
git revert 93ea0f972d591d54b07edba40a51b28486bbf9fd
这个时候,你期望发生什么呢?是不是 a.txt 的内容变成了:
a
c
d
然而并非如此,而是出现了冲突:
$ git revert 93ea0f972d591d54b07edba40a51b28486bbf9fd
Auto-merging a.txt
CONFLICT (content): Merge conflict in a.txt
error: could not revert 93ea0f9... fix: add b
hint: after resolving the conflicts, mark the corrected paths
hint: with 'git add <paths>' or 'git rm <paths>'
hint: and commit the result with 'git commit'
会变成了这样:
revert B 那一次操作会以 A 的状态为参考基准,即比较当前的状态和 A 状态的区别,发现冲突就会提示让用户自己去解决。
如果要撤销连续的提交,可以用一个范围来表示:
git revert 93ea0f972d591d54b07edba40a51b28486bbf9fd^..HEAD
这个时候你会发现记录变成下面这个样子了:
也就是说把 93ea0f972 和 HEAD 之间的所有提交都给按照顺序撤回了,这就是所谓的 range revert 语法:
git revert B^..D
这样就把 B,C,D 都给revert了,变成:
A-> B ->C -> D -> D'-> C' -> B'
这种方式,每次回滚都会生成一次新的提交。其实我们可以加个 -n
参数把多次合并成一次:
git revert -n 93ea0f972d591d54b07edba40a51b28486bbf9fd^..HEAD
再看提交记录:
是不是看起来就清爽多了,再复习一下语法:
git revert -n OLDER_COMMIT^..NEWER_COMMIT
git commit -m "revert OLDER_COMMIT to NEWER_COMMIT"