撤消 Git 中一个文件的工作副本修改?

在最后一次提交之后,我修改了工作副本中的一堆文件,但是我想撤消对这些文件之一的更改,例如将其重置为与最新提交相同的状态。

但是,我只想撤消仅一个文件的工作副本更改,而没有其他操作。

我怎么做?

答案

您可以使用

git checkout -- file

您可以在不使用--情况下执行此操作(如 nimrodm 所建议),但是如果文件名看起来像分支或标签(或其他修订标识符),则可能会感到困惑,因此最好使用--

您还可以签出文件的特定版本:

git checkout v1.2.3 -- file         # tag v1.2.3
git checkout stable -- file         # stable branch
git checkout origin/master -- file  # upstream master
git checkout HEAD -- file           # the version from the most recent commit
git checkout HEAD^ -- file          # the version before the most recent commit

只需使用

git checkout filename

这会将文件名替换为当前分支的最新版本。

警告:您的更改将被放弃 - 不保留任何备份。

git checkout <commit> <filename>

我今天使用这个是因为我意识到我的 favicon 在升级到 drupal 6.10 之前已经被覆盖了几次提交,因此我不得不将其恢复。这是我所做的:

git checkout 088ecd favicon.ico

如果文件已经被暂存(在编辑文件后发生 git add 等情况时发生)以取消暂存更改。

采用

git reset HEAD <file>

然后

git checkout <file>

如果尚未上演,请使用

git checkout <file>

如果只想撤消对该文件的先前提交更改,则可以尝试以下操作:

git checkout branchname^ filename

这将签出上一次提交之前的文件。如果您想再提交一些,请使用branchname~n表示法。

我已经通过 git bash 完成了:

(use "git checkout -- <file>..." to discard changes in working directory)

  1. Git 状态。 [因此,我们已经看到一个文件被修改了。]
  2. git checkout-index.html [我在 index.html 文件中进行了更改:
  3. git status [现在这些更改已删除]

在此处输入图片说明

我总是对此感到困惑,因此这里是一个提醒测试用例。假设我们有这个bash脚本来测试git

set -x
rm -rf test
mkdir test
cd test
git init
git config user.name test
git config user.email test@test.com
echo 1 > a.txt
echo 1 > b.txt
git add *
git commit -m "initial commit"
echo 2 >> b.txt
git add b.txt
git commit -m "second commit"
echo 3 >> b.txt

此时,更改尚未在缓存中git status ,因此git status为:

$ git status
On branch master
Changes not staged for commit:
  (use "git add <file>..." to update what will be committed)
  (use "git checkout -- <file>..." to discard changes in working directory)

    modified:   b.txt

no changes added to commit (use "git add" and/or "git commit -a")

如果从这一点来看,我们执行git checkout ,结果是这样的:

$ git checkout HEAD -- b.txt
$ git status
On branch master
nothing to commit, working directory clean

相反,如果执行git reset ,则结果为:

$ git reset HEAD -- b.txt
Unstaged changes after reset:
M   b.txt
$ git status
On branch master
Changes not staged for commit:
  (use "git add <file>..." to update what will be committed)
  (use "git checkout -- <file>..." to discard changes in working directory)

    modified:   b.txt

no changes added to commit (use "git add" and/or "git commit -a")

因此,在这种情况下 - 如果未进行更改,则git reset不会git checkout ,而git checkout覆盖更改。


现在,假设对上面脚本的最后更改是暂存 / 缓存的,也就是说,我们还在git add b.txt进行了git add b.txt

在这种情况下,此时的git status为:

$ git status
On branch master
Changes to be committed:
  (use "git reset HEAD <file>..." to unstage)

    modified:   b.txt

如果从这一点来看,我们执行git checkout ,结果是这样的:

$ git checkout HEAD -- b.txt
$ git status
On branch master
nothing to commit, working directory clean

相反,如果执行git reset ,则结果为:

$ git reset HEAD -- b.txt
Unstaged changes after reset:
M   b.txt
$ git status
On branch master
Changes not staged for commit:
  (use "git add <file>..." to update what will be committed)
  (use "git checkout -- <file>..." to discard changes in working directory)

    modified:   b.txt

no changes added to commit (use "git add" and/or "git commit -a")

因此,在这种情况下 - 如果更改是分阶段的,则git reset将基本上使分阶段的更改成为未分阶段的更改 - 而git checkout将完全覆盖更改。

该答案用于撤消相同或多个文件夹(或目录)中多个特定文件中的本地更改所需的命令。此答案专门解决了一个问题,即用户具有多个文件,但用户不想撤消所有本地更改:

如果您有一个或多个文件,可以通过列出每个文件的位置(用空格隔开),对每个git checkout -- file应用相同的命令( git checkout -- file ),如下所示:

git checkout -- name1/name2/fileOne.ext nameA/subFolder/fileTwo.ext

请注意 name1 / name2 / fileOne.ext 之间的空格 nameA / subFolder / fileTwo.ext

对于同一文件夹中的多个文件:

如果您碰巧需要放弃某个目录中所有文件的更改,请使用 git checkout,如下所示:

git checkout -- name1/name2/*

上面的星号具有撤消 name1 / name2 下该位置的所有文件的技巧。

并且,类似地,以下操作可以撤消多个文件夹的所有文件中的更改:

git checkout -- name1/name2/* nameA/subFolder/*

再次注意上面的 name1 / name2 / * nameA / subFolder / * 之间的空格。

注意:name1,name2,nameA,subFolder - 所有这些示例文件夹名称均表示相关文件可能所在的文件夹或软件包。

我使用 SHA ID 还原我的文件,我要做的是git checkout <sha hash id> <file name>

对我来说只有一个有效

git checkout -p filename

在此处输入图片说明