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

has*_*sen 1550 git version-control dvcs file undo

在最后一次提交之后,我在工作副本中修改了一堆文件,但我想撤消对其中一个文件的更改,就像将其重置为与最近提交的状态相同.

但是,我只想撤消单独一个文件的工作副本更改,而不是其它任何内容.

我怎么做?

Bri*_*ell 2109

您可以使用

git checkout -- file
Run Code Online (Sandbox Code Playgroud)

您可以在没有--(如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
Run Code Online (Sandbox Code Playgroud)

  • HEAD是当前分支上的最新提交,HEAD ^是当前分支上的提交.对于您描述的情况,您可以使用git checkout HEAD - filename. (62认同)
  • HEAD和HEAD ^之间有什么区别? (33认同)
  • 注意:如果文件已经暂存,则需要先重置它.`git reset HEAD <filename>; git checkout - <filename>` (29认同)
  • 简而言之,"git checkout sha-reference-filename",其中sha-reference是对提交的sha的引用,以任何形式(分支,标记,父级等) (15认同)
  • @gwho是的,你可以对最近的2次提交执行`HEAD ^^`,或者为3次提交执行`HEAD ^^^`.你也可以使用`HEAD~2`或`HEAD~3`,如果你想要更多的提交,它会更方便,而`HEAD ^ 2`意味着"这个提交的第二个父"; 因为合并提交,一个提交可以有多个以前的提交,所以使用`HEAD ^`一个数字选择那些父项,而使用`HEAD~`数字总是选择第一个父项但是提交的数量.有关详细信息,请参阅[`git help rev-parse`](https://www.kernel.org/pub/software/scm/git/docs/git-rev-parse.html). (13认同)

nim*_*odm 132

只是用

git checkout filename
Run Code Online (Sandbox Code Playgroud)

这将使用当前分支中的最新版本替换filename.

警告:您的更改将被丢弃 - 不保留备份.

  • @duckx是从文件名中消除分支名称的歧义.如果你说'git checkout x`和x恰好是一个分支名称和一个文件名,我不确定默认行为是什么,但我认为git会假设你想切换到分支x.当你使用`--`时,你说的是文件名. (19认同)
  • ic 感谢您解决这个问题。当他们向您展示示例时,每个人都假设您知道 - 意味着什么。它也不是你可以很容易地谷歌搜索的东西。 (2认同)
  • 看起来答案已被编辑以从中删除“--”。虽然仍然正确,但正如@hasen 指出的那样,如果文件名和分支名称之间存在歧义,您最终可能会出现非常不受欢迎的行为! (2认同)
  • 我喜欢它的方式,没有` - `,很好,很容易.当你使用文件名命名分支时,某些地方一定有不好的想法...... (2认同)

neo*_*eye 127

git checkout <commit> <filename>
Run Code Online (Sandbox Code Playgroud)

我今天使用了这个,因为我意识到当我升级到drupal 6.10时,我的favicon已被覆盖了几次提交,所以我不得不把它取回来.这是我做的:

git checkout 088ecd favicon.ico
Run Code Online (Sandbox Code Playgroud)

  • `git log --oneline <filename>`将为您提供更紧凑的日志,并且只包含对特定文件的更改 (5认同)
  • IMO通过命令行扫描gits日志并找到正确的文件很困难.使用GUI应用程序会更容易,例如http://www.sourcetreeapp.com/ (4认同)

tha*_*kal 66

如果您的文件已经暂存(当您在编辑文件后执行git add等操作时)会取消暂停您的更改.

使用

git reset HEAD <file>
Run Code Online (Sandbox Code Playgroud)

然后

git checkout <file>
Run Code Online (Sandbox Code Playgroud)

如果尚未上演,请使用

git checkout <file>
Run Code Online (Sandbox Code Playgroud)

  • 哈哈,这比被接受的哈哈更有帮助.很容易忘记已经上演的变化和没有变化的变化,因此重置有所帮助.虽然我之前也尝试过"git reset --hard",但它没有做"git reset HEAD"所做的事情.我想知道为什么? (2认同)

P-G*_*-Gn 29

Git 2.23 引入了 arestore来做到这一点,我认为是为了让这类问题的答案变得简单。

git restore [--] <pathspec>...
Run Code Online (Sandbox Code Playgroud)

与往常一样,--可能需要 ,但是当文件名以破折号开头时。(这里不可能与分支名称混淆,因为restore的周边不包括分支,这与 do-all 不同checkout

为了完整起见,restore还可以使用 恢复暂存文件,并从与--staged使用 不同的提交恢复。HEAD--source=<tree>

  • 也许对某些人来说这是显而易见的,但对我来说却不是:要恢复所有内容:“git Restore”。 (2认同)

syk*_*ora 19

如果您只想撤消先前提交对该文件的更改,您可以尝试这样做:

git checkout branchname^ filename
Run Code Online (Sandbox Code Playgroud)

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

  • 虽然如此,原始海报只是想恢复他的工作副本修改(我认为),而不是从上次提交中恢复更改.原始海报的问题有点不清楚,所以我可以理解这种混乱. (2认同)
  • 也许不是OP的用途,但我正在寻找如何用主副本覆盖我的分支 - 这在替换“branchname^”时效果非常好 (2认同)

Shi*_*ari 14

我通过git bash完成了:

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

  1. Git状态.[所以我们看到一个文件被修改过了.]
  2. git checkout - index.html [我在index.html文件中已更改:
  3. git status [现在删除了那些更改]

在此输入图像描述


sda*_*aau 8

我总是对此感到困惑,所以这里有一个提醒测试用例; 假设我们有这个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
Run Code Online (Sandbox Code Playgroud)

此时,更改不会在缓存中暂存,因此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")
Run Code Online (Sandbox Code Playgroud)

如果从这一点开始,git checkout结果如下:

$ git checkout HEAD -- b.txt
$ git status
On branch master
nothing to commit, working directory clean
Run Code Online (Sandbox Code Playgroud)

如果我们这样做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")
Run Code Online (Sandbox Code Playgroud)

因此,在这种情况下 - 如果更改没有暂存,git reset则没有区别,同时git checkout覆盖更改.


现在,让我们说上面脚本的最后一个更改是暂存/缓存,也就是说我们最后也做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
Run Code Online (Sandbox Code Playgroud)

如果从这一点开始,git checkout结果如下:

$ git checkout HEAD -- b.txt
$ git status
On branch master
nothing to commit, working directory clean
Run Code Online (Sandbox Code Playgroud)

如果我们这样做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")
Run Code Online (Sandbox Code Playgroud)

因此,在这种情况下 - 如果更改已暂存,git reset将基本上将分阶段更改为未分阶段的更改 - 同时git checkout将完全覆盖更改.


Nir*_*mal 7

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

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

git checkout -- name1/name2/fileOne.ext nameA/subFolder/fileTwo.ext
Run Code Online (Sandbox Code Playgroud)

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

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

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

git checkout -- name1/name2/*
Run Code Online (Sandbox Code Playgroud)

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

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

git checkout -- name1/name2/* nameA/subFolder/*
Run Code Online (Sandbox Code Playgroud)

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

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


Bet*_*eto 5

我使用SHA id恢复我的文件,我做的是什么 git checkout <sha hash id> <file name>


Ram*_*thi 5

对我来说只有这个有效

git checkout -p filename
Run Code Online (Sandbox Code Playgroud)

在此输入图像描述