仅使用git-restore<commit_hash>是行不通的。显然,必须指定-m。
当前回答
我从这个链接找到了如何恢复合并的很好的解释,我复制粘贴了下面的解释,如果下面的链接不起作用,这将很有帮助。
如何还原错误合并艾伦(alan@clueserver.org)他说:
我有一个主分支。我们有一个分支开发人员正在进行工作。他们声称它已经准备好了。我们将其合并进入主分支。它打破了一些东西,所以我们恢复合并。他们对代码进行更改。他们会说没关系,我们再次合并。当检查时,我们发现在还原之前所做的代码更改不在主分支中,但之后的代码更改位于主分支中树枝并请求帮助从这种情况中恢复过来。
“恢复合并”后的历史记录如下所示:
---o---o---o---M---x---x---W
/
---A---B
其中A和B处于不太好的侧开发中,M是将这些过早的更改引入主线的合并,x是与侧分支在主线上所做的和已经做的更改无关的更改,W是“合并M的恢复”(W看起来不是M颠倒了吗?)。IOW,“diff W^..W”类似于“diff-R M^..M”。
合并的“还原”可以通过以下方式进行:
$git还原-m 1 m在分支的开发人员纠正错误后,历史可能如下:
---o---o---o---M---x---x---W---x
/
---A---B-------------------C---D
其中C和D将修复A和B中的损坏,并且您可能已经在W之后的主线上进行了一些其他更改。
如果您合并更新后的分支(顶端有D),那么在A或B中所做的任何更改都不会产生结果,因为它们被W还原。这就是Alan看到的。
Linus解释了情况:
恢复常规提交只会有效地撤消该提交确实如此,而且相当简单。但同时恢复合并提交撤消提交更改的数据,但它确实与合并对历史的影响无关。因此,合并仍将存在,并且仍将被视为加入这两个分支合并在一起,将来的合并将看到最后一个共享状态-以及还原合并带来的还原不会影响这一点。因此,“还原”可以撤消数据更改,但它在很大程度上不是“撤销”是指它不会撤销提交对存储库历史记录。所以,如果你把“恢复”看作“撤消”,那么你将永远错过了这部分回复。是的,它撤销了数据,但不,它没有撤消历史记录。在这种情况下,您需要首先还原上一个还原,这将使历史看起来像这样:
---o---o---o---M---x---x---W---x---Y
/
---A---B-------------------C---D
其中Y是W的回复。这种“回复回复”可以通过以下方式完成:
$git还原W这段历史(忽略W和W..Y之间可能发生的冲突)相当于历史中根本没有W或Y:
---o---o---o---M---x---x-------x----
/
---A---B-------------------C---D
并且再次合并侧分支将不具有由较早的恢复和恢复的恢复引起的冲突。
---o---o---o---M---x---x-------x-------*
/ /
---A---B-------------------C---D
当然,在C和D中所做的更改仍然可能与任何x所做的操作发生冲突,但这只是正常的合并冲突。
其他回答
如果您想恢复合并提交,以下是您必须执行的操作。
首先,检查git日志以查找合并提交的id。您还可以找到与合并关联的多个父id(参见下图)。
记下黄色显示的合并提交id。父ID是在下一行中作为“合并:parent1 parent2”写入的ID。现在
短篇故事:
切换到进行合并的分支。然后,只需执行git-restore<mergecommit-id>-m1,它将打开vi控制台以输入提交消息。写,保存,退出,完成!
长话短说:
切换到进行合并的分支。在我的例子中,它是测试分支,我试图从中删除feature/analytics-v3分支。git-restore是还原任何提交的命令。但在恢复合并提交时有一个令人讨厌的技巧。您需要输入-m标志,否则它将失败。从这里开始,您需要决定是否要还原分支,并通过以下方式使其看起来与parent1或parent2完全相同:
git revert<merge commit id>-m 1(还原为parent2)
git revert<merge commit id>-m 2(还原为parent1)
你可以把这些父母记录下来,看看你想走哪条路,这是所有困惑的根源。
您可以按照以下步骤恢复不正确的提交或将远程分支重置回正确的HEAD/状态。
注意:此解决方案仅适用于您自己的分支,而不适用于共享的树枝
签出远程分支到本地回购。git签出您的branch_name从git日志中复制提交哈希(即错误提交前的提交id)git日志-n5
应该显示如下内容:
提交7cd42475d6f95f5896b6f02e902efab0b70e8038“合并分支“错误提交”到“your_branch_name”“提交”f9a734f8f44b0b37ccea769a2fd774c0f0c012“这是错误的提交”提交3779ab50e72908da92d2cfcd72256d7a09f444ba“这是正确的提交“
将分支重置为上一步中复制的提交哈希git reset<commit hash>(即3779ab50e72908da92d2cfcd72256d7a09f444ba)运行gitstatus以显示错误提交中的所有更改。只需运行gitreset——很难还原所有这些更改。强制将您的本地分支推到远程,并注意到您的提交历史记录与被污染之前一样干净。git push-f origin your_branch_name
我从这个链接找到了如何恢复合并的很好的解释,我复制粘贴了下面的解释,如果下面的链接不起作用,这将很有帮助。
如何还原错误合并艾伦(alan@clueserver.org)他说:
我有一个主分支。我们有一个分支开发人员正在进行工作。他们声称它已经准备好了。我们将其合并进入主分支。它打破了一些东西,所以我们恢复合并。他们对代码进行更改。他们会说没关系,我们再次合并。当检查时,我们发现在还原之前所做的代码更改不在主分支中,但之后的代码更改位于主分支中树枝并请求帮助从这种情况中恢复过来。
“恢复合并”后的历史记录如下所示:
---o---o---o---M---x---x---W
/
---A---B
其中A和B处于不太好的侧开发中,M是将这些过早的更改引入主线的合并,x是与侧分支在主线上所做的和已经做的更改无关的更改,W是“合并M的恢复”(W看起来不是M颠倒了吗?)。IOW,“diff W^..W”类似于“diff-R M^..M”。
合并的“还原”可以通过以下方式进行:
$git还原-m 1 m在分支的开发人员纠正错误后,历史可能如下:
---o---o---o---M---x---x---W---x
/
---A---B-------------------C---D
其中C和D将修复A和B中的损坏,并且您可能已经在W之后的主线上进行了一些其他更改。
如果您合并更新后的分支(顶端有D),那么在A或B中所做的任何更改都不会产生结果,因为它们被W还原。这就是Alan看到的。
Linus解释了情况:
恢复常规提交只会有效地撤消该提交确实如此,而且相当简单。但同时恢复合并提交撤消提交更改的数据,但它确实与合并对历史的影响无关。因此,合并仍将存在,并且仍将被视为加入这两个分支合并在一起,将来的合并将看到最后一个共享状态-以及还原合并带来的还原不会影响这一点。因此,“还原”可以撤消数据更改,但它在很大程度上不是“撤销”是指它不会撤销提交对存储库历史记录。所以,如果你把“恢复”看作“撤消”,那么你将永远错过了这部分回复。是的,它撤销了数据,但不,它没有撤消历史记录。在这种情况下,您需要首先还原上一个还原,这将使历史看起来像这样:
---o---o---o---M---x---x---W---x---Y
/
---A---B-------------------C---D
其中Y是W的回复。这种“回复回复”可以通过以下方式完成:
$git还原W这段历史(忽略W和W..Y之间可能发生的冲突)相当于历史中根本没有W或Y:
---o---o---o---M---x---x-------x----
/
---A---B-------------------C---D
并且再次合并侧分支将不具有由较早的恢复和恢复的恢复引起的冲突。
---o---o---o---M---x---x-------x-------*
/ /
---A---B-------------------C---D
当然,在C和D中所做的更改仍然可能与任何x所做的操作发生冲突,但这只是正常的合并冲突。
-m1是被修复的当前分支的最后一个父级,-m2是合并到其中的分支的原始父级。
如果命令行令人困惑,Tortoise Git也可以在这里提供帮助。
接受的答案和其他一些答案演示了如何使用git-restore命令还原合并提交。然而,对于父提交存在一些混淆。这篇文章旨在通过一个图形表示和一个真实的例子来阐明这一点。
还原合并提交不像git revert<commit hash>那样简单,因为git在从合并提交返回时会因为两个父提交而感到困惑。要指定所需的父级,请使用-m标志。由于git无法确定哪个父级是主线,哪个分支是要自动取消合并的分支,因此必须指定此项。
iss53分支被合并到master中,创建了一个合并提交C6。C6有两个父母C5和C4。
需要恢复C6并将存储库恢复到C4的状态。因此,它必须指定要用于revert命令的父级。
为此,请检查git日志(此处表示实际的提交哈希值和图中的代码名称)>git日志提交C6合并:C4 C5作者:Mozz<mozz@example.com>日期:2月29日星期三23:59:59 2020+0100将分支“iss53”合并到主分支...从git日志输出中,记下Merge:--附带的父ID。它的格式为Merge:parent1 parent2,此处为Merge:C4 C5。C4提交在主分支中,我们需要恢复到它,即父1,这里需要-m1(使用gitlogC4验证先前的提交以确认父分支)。切换到进行合并的分支(这里是主分支,我们的目标是从中删除iss53分支)使用-m 1标志执行git还原。#在主分支中恢复到C4数字还原C6-m 1#C6-是合并提交哈希
对于一些其他情况,
# revert to C5 in iss53 branch
git revert C6 -m 2
# General
git revert <merge commit id> -m 1 (reverts to parent1)
git revert <merge commit id> -m 2 (reverts to parent2)
# remember to check and verify the parent1 and parent2 with git log command.
实际示例
在只有主分支的现有项目上创建了一个新的分支还原测试,提交图现在看起来像这样。
(对于提交的图形化视图,请使用带有git-log[SO ans-ref]的-graph或这个更具交互性的VS代码扩展-git-grraph)
现在,我已经添加了一些新文件,修改了现有文件,并在每个分支上创建了单独的提交,然后将它们推送到源。图形现在看起来如下:
然后,从GitHub创建一个拉取请求,并将恢复测试分支合并到main。
我想撤消合并提交并返回到主分支中的最后一次提交,即12a7327
注意,合并提交-2ec06d9现在有两个父级-12a7327(在主级)和15bde47(在还原测试中),现在检查gitlog,
> git log
commit 2ec06d9d315a3f7919ffe4ad2c2d7cec8c8f9aa3 (HEAD -> main, origin/main, origin/HEAD)
Merge: 12a7327 15bde47
Author: Akshay <63786863+akshay@users.noreply.github.com>
Date: Sun Feb 5 00:41:13 2023 +0530
Merge pull request #1 from Akshay/revert-test
Revert test
要恢复合并提交并返回到12a7327需要执行的操作,
# To the First parent
git revert 2ec06d9 -m 1
现在,编辑器中将显示一条提交消息,指定详细信息、检查和验证。
因此,这将创建一个Revert提交,该提交执行合并提交的反向更改。
最后推送更改,现在合并提交更改消失,日志将如下所示:,
推荐文章
- 为什么我需要显式地推一个新分支?
- 如何撤消最后的git添加?
- Rubymine:如何让Git忽略Rubymine创建的.idea文件
- Gitignore二进制文件,没有扩展名
- Git隐藏错误:Git隐藏弹出并最终与合并冲突
- 了解Git和GitHub的基础知识
- 没有。Git目录的Git克隆
- Git与Mercurial仓库的互操作性
- 忽略git中修改(但未提交)的文件?
- “git restore”命令是什么?“git restore”和“git reset”之间有什么区别?
- Git合并与强制覆盖
- Git拉另一个分支
- 在Bash命令提示符上添加git分支
- 如何更改Git日志日期格式
- git pull -rebase和git pull -ff-only之间的区别