从Git提交中删除文件


1611

我正在使用Git,并且已经使用了一些文件

git commit -a

后来,我发现一个文件被错误地添加到提交中。

如何从上一次提交中删除文件?


2
这个环节是非常适合你的问题: stackoverflow.com/questions/307828/...
b3h3m0th

@CharlesB:是的,这是我的最后一次提交
Lolly

8
您是否将提交推送到服务器?
Paritosh Singh 2012年

4
我只是这样做:git reset filepath
felipekm

Answers:


3086

我认为这里的其他答案是错误的,因为这是将错误提交的文件从上一次提交移回到暂存区的问题,而不取消对它们所做的更改。可以像Paritosh Singh建议的那样完成:

git reset --soft HEAD^ 

要么

git reset --soft HEAD~1

然后重置不需要的文件,以使它们脱离提交:

git reset HEAD path/to/unwanted_file

现在再次提交,您甚至可以重复使用相同的提交消息:

git commit -c ORIG_HEAD  

86
谢谢你 值得补充的是,如果您已经推送了较早的(错误的)提交,现在又尝试git push修复您的存储库,它将抱怨Updates were rejected because the tip of your current branch is behind its remote counterpart.。如果您确定要推动它们(例如,这是您的叉子),则可以使用该-f选项来强制推动,例如git push origin master -f。(不要对其他人正在获取的上游
仓库

57
git reset --soft HEAD^是我最常用的撤消操作
Funroll

2
@PabloFernandez,首先,被接受的答案可能是OP所寻找的(此外,它是在几个月前发布的)。其次,无论投票数多少,公认的答案始终是最重要的。
MITjanitor

4
所有答案顶部的@PabloFernandez是三个选项卡,可让您控制答案的顺序:有效最旧投票。我的猜测是你设置为最古老的。即使接受的答案仍位于顶部,将其切换为投票,该答案将排在第二。
ahsteele

15
我对此非常了解,git reset但是希望找到一种方法来影响“到位”的现有提交。我刚刚了解到git commit -C。因此,对我来说,我想要的是您的精确配方,再走一步,将新的“再次提交”拼写为git commit -C [hash of original HEAD commit from first step]
metamatt 2014年

323

注意!如果您只想从先前的提交中删除文件,并将其保存在磁盘上,请阅读上面的juzzlin答案

如果这是您的最后一次提交,并且您想从本地和远程存储库中完全删除文件,则可以:

  1. 删除文件 git rm <file>
  2. 提交带有修改标志: git commit --amend

amend标志告诉git再次提交,但是将“ commit”(不是合并两个分支的意思)与最后一次提交“合并”。

如注释中所述,git rm此处使用就像使用rm命令本身!


120
您还可以使用git rm --cached将文件保留在磁盘上
Arkadiy Kukarkin

14
警告浏览此答案的人员:确保您要删除文件(就像走了走了一样!),而不仅仅是从“提交”列表中将其删除。
Scott Biggs 2015年

8
要添加到别人怎么说(并使其更容易记住一定不要这样做,除非你真的想):rmgit命令做什么的rm自己呢!
2016年

@CharlesB您能否将Arkadiy Kukarkin的注释中的注释添加到您的答案中,以使其更具可见性?
mopo922

2
请注意,如果您改变了主意,则仍可以还原文件,之前的提交git commit --amend仍然存在,并且可以使用进行查找git reflog。因此,它没有其他评论所建议的那样糟糕。
Steohan '17

165

现有的答案都在谈论从上一次提交中删除不需要的文件。

如果要从提交(甚至推送)中删除不需要的文件,并且不想创建新的提交,则由于该操作,这是不必要的:

1。

查找您要文件符合的提交。

git checkout <commit_id> <path_to_file>

如果要删除许多文件,可以多次执行此操作。

2。

git commit -am "remove unwanted files"

3。

找到错误添加了文件的提交的commit_id,在这里说“ 35c23c2”

git rebase 35c23c2~1 -i  // notice: "~1" is necessary

该命令根据您的设置打开编辑器。默认值是vim。

将最后一个提交(应为“删除不需要的文件”)移动到错误提交的下一行(在本例中为“ 35c23c2”),并将命令设置为fixup

pick 35c23c2 the first commit
fixup 0d78b28 remove unwanted files

保存文件后应该会很好。

完成 :

git push -f

如果不幸遇到冲突,则必须手动解决。


2
用nano做到这一点真是太好了!Ctrl + K,Ctrl + U,'f',Ctrl + X,'y',瞧!
sequielo '16

6
如果您想从存储库中实际删除文件(而不是文件系统),而不是仅将它们还原到以前的版本,请执行步骤1而不是git rm --cached <file(s)>
waldyrious

2
等等,您可以随意在交互式rebase文件中移动提交吗?
Dan Rosenstark '17

2
您完全可以,但是您可能(或可能不会)发生冲突。
布赖恩

6
通过添加--fixup=35c23c2git commit命令,可以使此过程更加容易。这将自动将提交设置为所需提交的修正,因此您无需在基准库中指定它。此外,如果您添加--autosquashgit rebase命令中,则git会自动将提交移至正确的位置,因此您无需在交互式基础库中做任何事情-只需保存结果即可(这意味着您甚至不需要-i标记,尽管我还是喜欢使用它来确保一切看起来都和我期望的一样)。
古斯

144

如已接受的答案所示,您可以通过重置整个提交来做到这一点。但这是一个相当繁重的做法。
一种更清洁的方法是保留提交,然后从中删除更改的文件。

git reset HEAD^ -- path/to/file
git commit --amend --no-edit

git reset将文件,因为它是在以前的承诺,并在索引阶段,。工作目录中的文件未更改。
然后,git commit它将提交并将索引压缩到当前提交中。

本质上,这将获取上一次提交中文件的版本,并将其添加到当前提交中。这不会导致任何净变化,因此可以有效地将文件从提交中删除。


5
对于只删除一个文件而不检查整个提交,这是一个更好的答案。
nimish

这与该答案完全相同:D stackoverflow.com/a/27340569/1623984
ThatsAMorais

@ThatsAMorais确实是:-)。我想知道这个问题是否与其他问题合并,这就是为什么我没有看到它。也许我只是瞎子。无论哪种情况,我都认为我倾向于将其保留为基于票数,似乎更受欢迎(也许人们更喜欢简短直接的回答)。
Patrick

一定要离开它!:)我们显然有正确的想法,目的是提供帮助。我认为合并是一个很好的理论。
ThatsAMorais

如果您已经推送到github上的一个分支,该怎么做?在完成此修复后,我尝试进行推送,并收到消息“更新被拒绝,因为您当前分支的提示位于提示之后:它的远程副本。”
奥利·威廉姆斯

41

如果尚未在服务器上推送更改,则可以使用

git reset --soft HEAD~1

它将重置所有更改并恢复为一次提交

如果您已推送更改,请按照@CharlesB回答的步骤进行操作


2
-1 git reset从临时区域中删除发生更改的文件,此处更改已提交
CharlesB 2012年

好的,但是我会继续投票,因为这不是OP所想要的:)对不起
CharlesB 2012年

对我来说还可以,但是为什么对手不想要这个。问题是什么?
Paritosh Singh 2012年

没什么大不了的,但是因为OP希望从上次提交中删除不需要的文件,并且它只是重置为提交前的状态。仍然需要重做提交。
CharlesB 2012年

3
@Aris使用<git diff --cached>查看更改
Paritosh Singh

37

使用rm删除文件将删除它!

您总是在git中添加提交而不是删除,因此在这种情况下,将文件返回到第一次提交之前的状态(如果是新文件,这可能是删除'rm'动作),然后重新提交,文件将消失。

要将文件恢复到以前的状态:

    git checkout <commit_id> <path_to_file>

或将其返回到远程HEAD的状态:

    git checkout origin/master <path_to_file>

然后修改提交,您应该会发现文件已从列表中消失(而不是从磁盘中删除!)



29

以下内容将仅取消暂存您想要的文件,这是OP要求的。

git reset HEAD^ /path/to/file

您会看到类似以下的内容...

要提交的更改:(使用“ git reset HEAD ...”取消登台)

修改:/ path / to / file

未上演提交的更改:(使用“ git add ...”更新将提交的内容)(使用“ git checkout-...”放弃工作目录中的更改)

修改:/ path / to / file

  • “要提交的更改”是提交之前文件的先前版本。如果文件从不存在,则看起来像是删除。如果您提交此更改,则将有一个修订版本,可将更改还原到分支中的文件。
  • “未暂存的更改未提交”是您已提交的更改以及文件的当前状态

此时,您可以对文件执行任何操作,例如重置为其他版本。

准备提交时:

git commit --amend -a

或(如果您正在进行其他更改,但还不想提交)

git commit add /path/to/file
git commit --amend

3
juzzlin的答案很好,但是当您只想撤消整个提交时,过多地撤消整个提交。如果您当前不想取消对该提交中的文件的更改,则对整个提交进行暂存可能会导致问题。
ThatsAMorais 2014年

27

我将通过示例向您解释。
令A,B,C为3次连续提交。提交B包含不应提交的文件。

git log  # take A commit_id
git rebase -i "A_commit_ID" # do an interactive rebase
change commit to 'e' in rebase vim # means commit will be edited
git rm unwanted_file
git rebase --continue
git push --force-with-lease <branchName>    

如果特定文件不在上一个或上一个提交中,这是最优雅的方法。我什至会说这是总体上最优雅的方式。我喜欢互动式基地。
bvgheluwe

对于单个提交,请更改noopedit [A_commit_ID]e [A_commit_ID]
TamusJRoyce

23

您可以尝试一下。

git reset --soft HEAD~1

并创建一个新的提交。

但是,有一个很棒的软件“ gitkraken”。这使得使用git变得容易。


1
请注意:在此之后,您git commit --amend需要在上一次提交中更新文件删除;事后,您可以检查它确实被去掉git log -1 --stat
sdbbs

13
git rm --cached <file_to_remove_from_commit_<commit_id>_which_added_file>
git commit -m "removed unwanted file from git"

会将您的本地文件保留下来。如果您也不希望在本地存储文件,则可以跳过--cached选项。

如果所有工作都在本地分支上,则需要在以后的提交中保留该文件,并且像拥有干净的历史记录一样,我认为一种更简单的方法可以是:

git rm --cached <file_to_remove_from_commit_<commit_id>_which_added_file>
git commit --squash <commit_id>
git add <file_to_remove_from_commit_<commit_id>_which_added_file>
git commit -m "brand new file!"
git rebase --interactive <commit_id>^

然后您就可以轻松完成变基,而不必记住更复杂的命令或提交消息或键入太多内容。


这对我有用。我假设如果您想将文件重新添加到混合中,只需使用git add -A或git add即可。他们回来了。
亚历山大·米尔斯

11

使用git GUI可以简化从先前提交中删除文件的过程。

假设这不是共享分支,并且您不介意重写history,请运行:

git gui citool --amend

您可以取消选中错误提交的文件,然后单击“提交”。

在此处输入图片说明

该文件已从提交中删除,但将保留在磁盘上。因此,如果您在错误添加文件后取消选中该文件,它将显示在未跟踪的文件列表中(如果您在错误修改文件后取消选中了该文件,它将显示在未暂存的提交更改中)。


2
在Ubuntu上,您可以使用sudo apt-get install git-gui
JDiMatteo17

谢谢!我陷入了一个问题(错误?),其中添加了一个包含.git repo的文件夹,而所有常规的remove命令都无法正常工作。但是,这有所帮助。回过头来,我先利用了它git rebase -i HEAD~4,然后运行了您的命令以打开编辑器。另一个注意事项:在“提交”菜单中可以找到“取消登台”。
Johny Skovdal

最简单的解决方案。最简单的记住。与使用git reset --soft HEAD^(记住--soft arg),然后使用git commit -c ORIG_HEAD(而不是--amend,将所有内容搞砸了)相比,出错的可能性要小得多。
布伦特·浮士德·浮士德

9

如果您想保留提交(也许您已经花了一些时间编写详细的提交消息并且不想丢失它),并且只想从提交中删除文件,而不是从存储库中完全删除:

git checkout origin/<remote-branch> <filename>
git commit --amend

6

执行以下命令序列:

//to remove the last commit, but preserve changes  
git reset --soft HEAD~1

//to remove unneded file from the staging area  
git reset HEAD `<your file>` 

//finally make a new commit  
git commit -m 'Your message'

我尝试执行这些步骤,但看到此错误错误:无法将一些引用推送到“ git ...”。为防止丢失历史记录,拒绝了非快进更新,并合并了远程更改(例如'git pull'),然后再次推送。有关详细信息,请参见“ git push --help”的“关于快进的说明”部分。(在git pull之后,我有相同的更改)
Dezigo 2015年

这意味着在执行本地工作时,远程回购的状态已更改。在“ git pull”之后,您的本地更改应与远程更改合并,仅此而已。当然,您的更改必须保留。
谢尔盖·奥尼申科

换句话说,如果收到该错误@Dezigo,请添加-f标志以强制更新。
jungledev

5

只是想补充最重要的答案,因为我必须运行一个额外的命令:

git reset --soft HEAD^
git checkout origin/master <filepath>

干杯!


欢迎。如果您解释了命令的实际作用,则此答案会更好。
Mark Chorley

3

对我有用的东西,但仍然认为应该有一个更好的解决方案:

$ git revert <commit_id>
$ git reset HEAD~1 --hard

只需将要放弃的更改留在另一次提交中,然后检查其他人

$ git commit --amend // or stash and rebase to <commit_id> to amend changes

3

git reset --soft HEAD^回退您的提交,当您键入时git status,它会告诉您该怎么做:

Changes to be committed:
  (use "git reset HEAD <file>..." to unstage)

2

实际上,我认为一种更快,更轻松的方法是使用git rebase交互模式。

git rebase -i head~1  

(或头〜4,您想走多远)

然后使用“编辑”代替“选择”。我没有意识到“编辑”有多么强大。

https://www.youtube.com/watch?v=2dQosJaLN18

希望对您有所帮助。


该视频只有10分钟,而且用处
不大

2

在我只想还原一个文件的本地分支中进行更改时,发生了相同的问题。对我有用的是-

(下面的feature / target_branch是我进行所有更改的地方,包括我想撤消特定文件的更改)

origin / feature / target_branch是要将更改推送到的远程分支)

功能/阶段是我的临时阶段分支,我将从那里进行所有想要的更改(不包括对该文件的更改)

  1. 从我的origin / feature / target_branch创建一个本地分支-称为feature / staging

  2. 将我正在工作的本地分支Feature / target_branch合并feature / staging分支

  3. 检出功能部件/暂存,然后git reset --soft ORIG_HEAD (现在,功能暂存中的所有更改都将暂存,但未提交。)

  4. 取消暂存我先前签入的文件并进行不必要的更改

  5. 功能/阶段的上游分支更改为 origin / feature / target_branch

  6. 提交了其余的分阶段更改,并向上游推送到我的远程origin / feature / target_branch


1

如果您不再需要该文件,则可以

git rm file
git commit --amend
git push origin branch

1

如果您使用的是GitHub且尚未推送提交,则GitHub Desktop可以轻松解决此问题:

  1. 选择存储库->撤消最近提交
  2. 取消选择您错误添加的文件。您先前的提交消息将已经在对话框中。
  3. 按下提交按钮!

1

如果要从以前的提交中删除文件,请使用过滤器

git filter-branch --prune-empty --index-filter 'git rm --ignore-unmatch --cached "file_to_be_removed.dmg"'

如果看到此错误:

无法创建新的备份。refs / original /中已经存在先前的备份,请使用-f强制覆盖备份

只需删除本地仓库上的裁判备份

$ rm -rf .git/refs/original/refs

1

如果您不将更改推送到git

git reset --soft HEAD~1

它将重置所有更改并恢复为一次提交

如果这是您最后一次提交,并且要从本地和远程存储库中删除文件,请尝试以下操作:

git rm <file>
 git commit --amend

甚至更好:

首先重置

git reset --soft HEAD~1

重置不需要的文件

git reset HEAD path/to/unwanted_file

再次提交

git commit -c ORIG_HEAD  

与上述相同,但甚至确实可以进行交叉检查
Reshma

0

这对我来说是从位存储库回购中删除文件的,我最初将文件推送到分支中。

git checkout origin/develop <path-to-file>
git add <path-to-file>
git commit -m "Message"
git push

0

我将当前文件复制到另一个文件夹中,然后通过以下方法摆脱所有未进行的更改:

git reset --hard @{u}

然后将内容复制回去。提交,推送。


0

您可以简单地使用以下命令:

git restore --staged <file>

0
Here is the step to remove files from Git Commit.

>git reset --soft HEAD^1(either commitid ) -- now files moved to the staging area.
>git rm --cached filename(it will removed the file from staging area)
>git commit -m 'meaningfull message'(Now commit the required files)

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.