在进度落后的项目中增加人手只会导致进度更加落后
--Fred Brooks (图灵奖得主 《人月神话》作者)

掌握Git撤销操作,随心所欲控制文件状态

通过前面两篇 git 的相关介绍(花10分钟开始使用Git,和精通Git不得不了解的基本概念),相信此时你应该能够使用 git 轻松应付一般性工作了,前面的讨论中,我们留下了一个问题:执行 git add 暂存文件之后,如何撤销其暂存状态?

本文主要讨论和撤销有关的 git 操作。目的是让读者在遇到关于撤销问题时能够方便迅速对照执行解决问题,而不用去翻阅参数繁多的 git 使用说明。

一开始你只需了解大致功能即可,不必记住所有命令和具体参数。事实上,如果没有经过反复多次的操作,这些没血没肉的命令是很难被全部记清楚的,就算现在记住了,也会很快遗忘(天赋异禀,过目不忘者除外)。建议读者在遇到特定问题时对照场景操作,多用几次自然就记住了。

首先,我们再看看上一篇文章中的这张图:

git.001
git 命令和文件状态转换
 

这张图能让你一目了然的看到各种命令产生的效果。其中箭头移动方向可以理解成文件版本的复制方向。根据箭头的指向,我们能清楚的看到每个操作所产生的影响。几乎所有的 git 操作也就是让文件在这三个工作区域内移动,列如:

  • git add files 工作目录下files复制 ->暂存区;
  • git checkout files 暂存区files复制 -> 工作目录;
  • git commit 暂存区内的文件作为一个版本保存 -> 版本库。

为了使本文更具实用性,让我们来结合上图,通过实际项目中可能遇到的问题,以 Q&A 的形式来逐一讨论,当你遇到相似问题时可以迅速找到解决方法。

  1. Q: git add 之后如何撤销?
    TL;DR[1]答案: 可以使用 git reset 撤销所有暂存区域文件。使用 git reset file 来撤销特定的文件。该命令只是把暂存区内容移除,不会覆盖工作目录中已经修改过的同名文件。

    考虑这个场景,你正在为你的项目添加一个新功能,快完成时,你打算暂存起来,测试好了再 commit,于是你使用:
    git add home.js
    把涉及新功能改动的文件 home.js 暂存起来。此时你突然接到一个很紧急的 bug,需要立刻修复,你在一通 debug 之后成功 fix 了 bug,于是把涉及该 bug 的更新添加到暂存区:
    git add main.js common.js

    在准备使用 git commit 提交改动之前,你习惯性看看暂存区有哪些文件会被提交,于是你使用 git status 发现暂存区有三个文件:
    home.js main.js common.js

    很显然,你此时只想提交关于 bug 的修改部分文件,如果此时使用 git commit 会把关于其他功能的修改一起提交。那如何移走暂存区域 home.js?很简单,只要执行git reset即可移除:
    git reset home.js

    之后执行 git commit 来提交这次 bug 修复所引起的改动。

    当然,如果你打算先提交新功能改动之后再提交 bug 修复内容,你可以先使用
    git reset main.js common.js取消暂存区 bug 修改相关文件,再执行
    git commit

  2. Q: 如何丢弃工作目录的更改?
    答案:使用 git checkout file注意,该操作不可逆,一旦被执行,你的改动都会消失,一般情况下只有你非常确定不需要工作目录下的最新改动内容才会这么做。 该命令会用暂存区的file 内容覆盖掉工作目录的file 文件内容。如果这个文件没有加入暂存区(或者说暂存区该 file 内容和 Git 仓库内容相同),执行该命令后,工作目录中该文件版本将和 Git 仓库最近一次提交版本相同。举例:
    git checkout home.js

    注意,如果你正好有个 branch 名字叫 home.js, git checkout 会checkout 一个 branch,此时你可以加上 -- 来用来表明这是个文件。
    git checkout -- home.js

    如果你想直接用最新的 git 仓库文件同时覆盖掉暂存区和工作目录内容可以使用(参见前文图示):
    git checout HEAD -- files

  3. Q: 如何丢弃本地仓库的 commit 内容?
    git reset --soft HEAD~1
    执行上述操作会撤销最后一次 commit 产生的效果并且保留 working directory 你所改动的内容。

    git reset --hard HEAD~1
    执行上述操作会撤销最后一次 commit 产生的效果并且覆盖 working directory 你所改动的内容。慎用,你一天的工作可能就此付之一炬,出了问题不能懒我(无辜状)。

    如果想取消最近N次的 commit 内容,只要把上面的1换成相应的数字即可。

  4. Q: 如何获取Git服务器上最新内容?
    经常拉取新内容,可以减少 code merge。只要简单的执行 git pull 即可。注意还有一个类似的命令是git fetch,二者的区别简单来说是:
    git pull = git fetch + git merge

最后总结关于撤销操作的要点

操作 撤销
git add file git reset
重置暂存区内容,使得暂存区版本和仓库版本一致

git reset file
重置某一个文件的暂存状态,产生效果也就是把该文件从暂存区移除

修改了工作区文件 git checkout -- files
用暂存区版本覆盖工作区文件

git checout HEAD -- files
用git仓库最近版本覆盖工作区文件

git commit git reset --soft HEAD~1
撤销最近一次 commit 且不删除工作区改动

git reset --hard HEAD~1
撤销最近一次commit且强制同步工作区版本到这 commit 改动之前状态

参考文献

  1. Pro Git book
  2. A Visual Git Reference 须自备梯子

[1]:网络用语too long; did not read. 一般用于表达不想看的长篇大论。 ↩︎

 

版权声明
本博客所有文章皆为原创,作者保留所有版权。转载必须保证全文完整和包含本声明,并以超链接形式注明出处 http://www.macode.net/git-undo/

发表评论

电子邮件地址不会被公开。 必填项已用*标注