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

Undo working copy modifications of one file in Git

提问人:hasen 提问时间:3/28/2009 最后编辑:Peter Mortensenhasen 更新时间:7/1/2023 访问量:922982

问:

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

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

我该怎么做?

git 文件 版本控制 DVCS 撤消

评论


答:

26赞 sykora 3/28/2009 #1

如果您只想撤消上一次提交对该文件的更改,您可以尝试以下操作:

git checkout branchname^ filename

这将检出上次提交前的文件。如果您想再进行几次提交,请使用表示法。branchname~n

评论

1赞 FernandoEscher 2/28/2013
这不会从提交中删除更改,它只会将差异应用于 HEAD 上的版本。
2赞 5/30/2014
虽然是真的,但原始发帖人只是想恢复他的工作副本修改(我认为),而不是恢复上次提交的更改。原发帖人的问题有点不清楚,所以我可以理解这种困惑。
2赞 Alex 3/15/2019
也许不是 OP 的用途,但我正在寻找如何用 master 的副本覆盖我的分支 - 这在替换时效果很好branchname^
212赞 nimrodm 3/28/2009 #2

只需使用

git checkout filename

这会将 filename 替换为当前分支中的最新版本。

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

评论

29赞 hasen 3/5/2015
@duckx是为了消除分支名称与文件名的歧义。如果你说 x 恰好是分支名称和文件名,我不确定默认行为是什么,但我认为 git 会假设你想切换到分支 x。当您使用时,您说后面是文件名。git checkout x--
3赞 Patoshi パトシ 3/5/2015
IC感谢您澄清这一点。每个人都只是假设你知道什么 - 当他们向你展示例子时意味着什么。而且你也不容易用谷歌搜索它。
3赞 BrainSlugs83 3/15/2015
看起来答案已被编辑以从中删除。虽然仍然正确,但正如 @hasen 所指出的,如果文件名和分支名称之间存在歧义,您最终可能会在这里出现非常不受欢迎的行为!--
4赞 Marco Faustinelli 4/22/2017
我喜欢它的方式,没有,很好,很容易。当你用文件名命名分支时,一定有不好的想法......--
2658赞 Brian Campbell 3/28/2009 #3

你可以使用

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

评论

44赞 hasen 3/29/2009
HEAD 和 HEAD^ 有什么区别?
74赞 Paul 3/29/2009
HEAD 是当前分支上的最新提交,HEAD^ 是当前分支上之前的提交。对于您描述的情况,您可以使用 git checkout HEAD -- filename。
19赞 lprsd 3/2/2010
简而言之,“git checkout sha-reference -- filename”,其中 sha-reference 是对提交的 sha 的引用,以任何形式(分支、标签、父级等)
36赞 Olie 6/14/2013
注意:如果文件已暂存,则需要先重置它。git reset HEAD <filename> ; git checkout -- <filename>
14赞 Brian Campbell 5/13/2014
@gwho 是的,您可以执行最近的 2 次提交,也可以执行 3 次提交。你也可以使用 ,或者 ,如果你想返回更多的提交,它会更方便,而 表示“这个提交的第二个父级”;由于合并提交,一个提交可以有多个先前的提交,因此 with a number 选择其中的哪一个父级,而 with a number 始终选择第一个父级,但该数量的提交数会返回。有关更多详细信息,请参阅 git help rev-parseHEAD^^HEAD^^^HEAD~2HEAD~3HEAD^2HEAD^HEAD~
155赞 neoneye 3/28/2009 #4
git checkout <commit> <filename>

我今天使用它是因为我意识到当我升级到 drupal 6.10 时,我的网站图标在几次提交前被覆盖了,所以我必须把它拿回来。这是我所做的:

git checkout 088ecd favicon.ico

评论

1赞 Alex 3/1/2012
除了滚动抛出大量“git log --stat”输出之外,我如何获得(以前删除的文件)提交?
4赞 neoneye 3/2/2012
IMO:通过命令行扫描 gits 日志并找到正确的文件有点困难。使用 GUI 应用程序(例如 sourcetreeapp.com)要容易得多
6赞 rjmunro 2/5/2014
git log --oneline <filename>将为您提供更紧凑的日志,并且仅包含对特定文件的更改
1赞 ygesher 8/5/2015
或者,您可以使用git reflog <filename>
0赞 Dmitry Melnikov 11/7/2022
这是一个普遍的答案。如果已经提交了不需要的内容,那么大多数其他答案将不起作用,因为它们的目标是从上次提交中恢复文件内容。
5赞 Beto 5/23/2012 #5

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

100赞 thanikkal 7/3/2012 #6

如果您的文件已经暂存(在编辑文件后执行 git add 等操作时发生)以取消暂存更改。

git reset HEAD <file>

然后

git checkout <file>

如果尚未暂存,只需使用

git checkout <file>

评论

3赞 Arman Bimatov 10/6/2014
这比公认的更有帮助哈哈。很容易忘记哪些更改已经上演,哪些没有上演,因此重置会有所帮助。虽然我之前也尝试过“git reset --hard”,但它没有做“git reset HEAD”所做的事情。我想知道为什么?
1赞 Jesse Glick 5/1/2014 #7

如果您尚未推送或以其他方式共享您的提交:

git diff --stat HEAD^...HEAD | \
fgrep filename_snippet_to_revert | cut -d' ' -f2 | xargs git checkout HEAD^ --
git commit -a --amend
9赞 sdaau 2/9/2016 #8

我总是对此感到困惑,所以这里有一个提醒测试用例;假设我们有这个脚本来测试:bashgit

set -x
rm -rf test
mkdir test
cd test
git init
git config user.name test
git config user.email [email protected]
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
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 resetgit 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

如果从这一点开始,我们这样做,结果是这样的: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 resetgit checkout

12赞 Nirmal 1/24/2017 #9

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

如果您有一个或多个文件,则可以将相同的命令 ( ) 应用于 每个文件都列出它们的每个位置,以 空间如下: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 - 所有这些示例文件夹名称都表示相关文件可能驻留的文件夹或包。

0赞 Gina 4/27/2017 #10

如果已提交,您可以还原文件的更改并再次提交,然后使用上次提交压缩新提交。

评论

1赞 Adrian 4/27/2017
添加要使用的特定命令将有助于原始海报和未来的访问者。
5赞 Ramesh Bhupathi 8/19/2017 #11

对我来说,只有这个有效

git checkout -p filename

enter image description here

36赞 Py-Coder 4/10/2018 #12

我已经通过 git bash 完成了:

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

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

enter image description here

-4赞 Gene 12/31/2018 #13
git checkout a3156ae4913a0226caa62d8627e0e9589b33d04c -p */SearchMaster.jsp

故障: a3156ae4913a0226caa62d8627e0e9589b33d04c = 这是提交的哈希值。它在我自己的个人分支(不是主人)上。

-p 标志是路径。

*/SearchMaster,.jsp 是文件名。

28赞 P-Gn 4/12/2021 #14

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

git restore [--] <pathspec>...

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

为了完成,还可以使用 恢复暂存文件,并从与 不同的提交中恢复。restore--stagedHEAD--source=<tree>

评论

2赞 Vicente Garcia 4/19/2022
也许对某些人来说很明显,但对我来说却不是:只是恢复一切:git restore .
1赞 eric 5/11/2023
这是现代的正确答案
4赞 brethvoice 7/19/2022 #15

git checkout是这样做的旧方法。2020 年更新后,首选方式是:

git restore file.txt

请参阅此重复答案(请务必向下滚动到 2020 年更新):https://stackoverflow.com/a/31281748/12763497