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

时间:2009-03-28 05:09:03

标签: git file version-control dvcs undo

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

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

我该怎么做?

14 个答案:

答案 0 :(得分:2015)

您可以使用

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

答案 1 :(得分:125)

git checkout <commit> <filename>

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

git checkout 088ecd favicon.ico

答案 2 :(得分:122)

只需使用

git checkout filename

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

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

答案 3 :(得分:64)

如果您的文件已经暂存(在编辑文件后进行git add等时发生),则取消暂停您的更改。

使用

git reset HEAD <file>

然后

git checkout <file>

如果尚未上演,请使用

git checkout <file>

答案 4 :(得分:18)

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

git checkout branchname^ filename

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

答案 5 :(得分:11)

我已经完成了git bash:

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

  1. Git状态。 [所以我们看到一个文件被修改过了。]
  2. git checkout - index.html [我在index.html文件中已更改:
  3. git status [现已更改这些更改]
  4. enter image description here

答案 6 :(得分: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

此时,更改未在缓存中暂存,因此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 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将完全覆盖更改。

答案 7 :(得分:6)

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

  

如果您有一个或多个文件,则可以应用相同的命令(git checkout -- file)   每个文件通过列出它们各自的位置分隔   空间如:

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

请注意name1 / name2 / fileOne.ext nameA / subFolder / fileTwo.ext

之间的空格

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

  

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

git checkout -- name1/name2/*

上面的星号可以在name1 / name2下撤消该位置的所有文件。

  

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

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

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

注意:name1,name2,nameA,subFolder - 所有这些示例文件夹名称都指示了有问题的文件所在的文件夹或包。

答案 8 :(得分:5)

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

答案 9 :(得分:2)

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

git diff --stat HEAD^...HEAD | \
fgrep filename_snippet_to_revert | cut -d' ' -f2 | xargs git checkout HEAD^ --
git commit -a --amend

答案 10 :(得分:1)

对我来说只有这个工作

git checkout -p filename

enter image description here

答案 11 :(得分:0)

如果它已经提交,你可以恢复文件的更改并再次提交,然后用最后一次提交压缩新提交。

答案 12 :(得分:0)

我不知道为什么,但是当我尝试输入代码时,它会显示为图像。

enter image description here

答案 13 :(得分:0)

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

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

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

完整地说,restore 还可以使用 --staged 恢复暂存文件,并使用 HEAD 从不同于 --source=<tree> 的提交中恢复。