Отменить рабочую копию изменений одного файла в Git? - PullRequest
1471 голосов
/ 28 марта 2009

После последнего коммита я изменил кучу файлов в моей рабочей копии, но я хочу отменить изменения в одном из этих файлов, как при сбросе его в то же состояние, что и самый последний коммит.

Однако я хочу отменить изменения рабочей копии только одного этого файла, ничего больше с ним.

Как мне это сделать?

Ответы [ 13 ]

2024 голосов
/ 28 марта 2009

Вы можете использовать

git checkout -- file

Вы можете сделать это без -- (как предлагает nimrodm), но если имя файла выглядит как ветвь или тег (или другой идентификатор ревизии), оно может запутаться, поэтому лучше использовать --. 1006 *

Вы также можете проверить конкретную версию файла:

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
125 голосов
/ 28 марта 2009
git checkout <commit> <filename>

Я использовал это сегодня, потому что понял, что мой favicon был перезаписан несколько коммитов назад, когда я обновился до drupal 6.10, поэтому мне пришлось вернуть его. Вот что я сделал:

git checkout 088ecd favicon.ico
123 голосов
/ 28 марта 2009

Просто используйте

git checkout filename

Это заменит имя файла последней версией из текущей ветви.

ВНИМАНИЕ: ваши изменения будут отменены & mdash; резервная копия не сохраняется.

64 голосов
/ 03 июля 2012

Если ваш файл уже подготовлен (происходит, когда вы делаете git add и т. Д. После редактирования файла), чтобы отменить внесение изменений.

Использование

git reset HEAD <file>

Тогда

git checkout <file>

Если это еще не постановка, просто используйте

git checkout <file>
18 голосов
/ 28 марта 2009

Если вы хотите просто отменить изменения предыдущего коммита в этом файле, попробуйте следующее:

git checkout branchname^ filename

Это извлечет файл таким, каким он был до последнего коммита. Если вы хотите вернуться еще на несколько коммитов, используйте обозначение branchname~n.

11 голосов
/ 10 апреля 2018

Я сделал через git bash:

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

  1. Состояние Git. [Итак, мы видели один файл, модифицированный.]
  2. git checkout - index.html [я изменился в файле index.html:
  3. git status [теперь эти изменения были удалены]

enter image description here

8 голосов
/ 09 февраля 2016

Я всегда путаюсь с этим, так что вот контрольный пример с напоминанием; скажем, у нас есть этот 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 полностью перезапишет изменения.

6 голосов
/ 24 января 2017

Этот ответ предназначен для команды, необходимой для отмены локальных изменений, которые находятся в нескольких конкретных файлах в одной или нескольких папках (или каталогах). Этот ответ, в частности, отвечает на вопрос, где у пользователя более одного файла, но он не хочет отменять все локальные изменения:

если у вас есть один или несколько файлов, вы можете применить ту же команду (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/*

Звездочка, указанная выше, выполняет процедуру удаления всех файлов в этом месте с именем name / name2.

И, аналогично, следующее может отменить изменения во всех файлах для несколько папок:

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

снова обратите внимание на пробел между name1 / name2 / * nameA / subFolder / * в выше.

Примечание: name1, name2, nameA, subFolder - все эти примеры имен папок указывают папку или пакет, где могут находиться рассматриваемые файлы.

5 голосов
/ 23 мая 2012

Я восстанавливаю свои файлы, используя идентификатор SHA, что я делаю git checkout <sha hash id> <file name>

2 голосов
/ 19 августа 2017

У меня только этот работал

git checkout -p filename

enter image description here

...