我在我的私房钱里存了一小块。我已经使用git stash apply将它应用到我的工作副本。现在,我想通过反向应用补丁来撤销这些更改(有点像git revert会做的事情,但针对的是stash)。

有人知道怎么做吗?

澄清:在我的工作副本中有其他更改。我的具体情况很难描述,但您可以想象在存储库中有一些调试或实验代码。现在,它混合在我的工作副本与其他一些变化,我想看看效果与不从隐藏的变化。

目前看来stash还不支持这个功能,但是git的stash apply—reverse将是一个不错的功能。


当前回答

这早就应该了,但如果我正确地解释这个问题,我已经找到了一个简单的解决方案,注意,这是我自己的术语解释:

Git stash [save]将保存当前的更改,并将当前分支设置为“清洁状态”

git的存储列表提供如下内容

Git应用stash@{0}将当前分支设置为之前的stash [save]

Git结帐。将当前分支设置为存储[保存]之后

保存在stash中的代码不会丢失,它可以通过git apply stash@{0}再次找到。

不管怎样,这对我很管用!

其他回答

除了@Greg Bacon的答案之外,如果二进制文件被添加到索引中,并且是存储使用的一部分

git stash show -p | git apply --reverse

可能导致

error: cannot apply binary patch to '<YOUR_NEW_FILE>' without full index line
error: <YOUR_NEW_FILE>: patch does not apply

add -binary解决了这个问题,但不幸的是还没有找到原因。

 git stash show -p --binary | git apply --reverse

git stash[save]获取你的工作目录状态和索引状态,并将它们存储起来,将索引和工作区域设置为HEAD版本。

Git stash apply会带回这些更改,所以Git reset—hard会再次删除它们。

Git stash pop带回这些更改并删除顶部存储的更改,因此Git stash [save]将在这种情况下返回到之前(预弹出)状态。

git stash show -p | git apply --reverse

警告,这不会在所有情况下:“git apply -R”(man)不能正确处理两次触及同一路径的补丁,这已在git 2.30(2021年第一季度)中得到纠正。

这在将路径从常规文件更改为符号链接(反之亦然)的补丁中是最相关的。

参见Jonathan Tan (jhowtan)提交b0f266d(2020年10月20日)。 (由Junio C Hamano - gitster -在commit c23cd78中合并,2020年11月2日)

apply:当-R时,也是反向的section列表 协助:Junio C Hamano 署名:乔纳森·谭

A patch changing a symlink into a file is written with 2 sections (in the code, represented as "struct patch"): firstly, the deletion of the symlink, and secondly, the creation of the file. When applying that patch with -R, the sections are reversed, so we get: (1) creation of a symlink, then (2) deletion of a file. This causes an issue when the "deletion of a file" section is checked, because Git observes that the so-called file is not a file but a symlink, resulting in a "wrong type" error message. What we want is: (1) deletion of a file, then (2) creation of a symlink. In the code, this is reflected in the behavior of previous_patch() when invoked from check_preimage() when the deletion is checked. Creation then deletion means that when the deletion is checked, previous_patch() returns the creation section, triggering a mode conflict resulting in the "wrong type" error message. But deletion then creation means that when the deletion is checked, previous_patch() returns NULL, so the deletion mode is checked against lstat, which is what we want. There are also other ways a patch can contain 2 sections referencing the same file, for example, in 7a07841c0b ("git-apply: handle a patch that touches the same path more than once better", 2008-06-27, Git v1.6.0-rc0 -- merge). "git apply -R"(man) fails in the same way, and this commit makes this case succeed. Therefore, when building the list of sections, build them in reverse order (by adding to the front of the list instead of the back) when -R is passed.

我自己也有类似的问题,我认为你所需要做的就是git重置——很难,你不会失去你的变化或任何未跟踪的变化。

如果你阅读git stash——help中的文档,它声明apply是“像pop一样,但不从stash列表中删除状态”,所以状态仍然驻留在那里,你可以取回它。

或者,如果没有冲突,您可以在测试更改后再次git stash。

如果你确实有冲突,不要担心,git重置-hard不会失去他们,因为 “应用状态可能会因冲突而失败;在这种情况下,它不会从收藏列表中删除。你需要手动解决冲突,然后手动调用git stash drop。”

根据git-stash手册,“一个stash被表示为一个提交,它的树记录了工作目录的状态,它的第一个父级是创建stash时在HEAD处的提交,”git stash show -p给我们“记录在stash中的更改,作为隐藏状态与其原始父级之间的差异”。

为了保持其他更改不变,使用git stash show -p | patch——reverse如下所示:

$ git init
Initialized empty Git repository in /tmp/repo/.git/

$ echo Hello, world >messages

$ git add messages

$ git commit -am 'Initial commit'
[master (root-commit)]: created 1ff2478: "Initial commit"
 1 files changed, 1 insertions(+), 0 deletions(-)
 create mode 100644 messages

$ echo Hello again >>messages

$ git stash

$ git status
# On branch master
nothing to commit (working directory clean)

$ git stash apply
# On branch master
# Changed but not updated:
#   (use "git add <file>..." to update what will be committed)
#   (use "git checkout -- <file>..." to discard changes in working directory)
#
#       modified:   messages
#
no changes added to commit (use "git add" and/or "git commit -a")

$ echo Howdy all >>messages

$ git diff
diff --git a/messages b/messages
index a5c1966..eade523 100644
--- a/messages
+++ b/messages
@@ -1 +1,3 @@
 Hello, world
+Hello again
+Howdy all

$ git stash show -p | patch --reverse
patching file messages
Hunk #1 succeeded at 1 with fuzz 1.

$ git diff
diff --git a/messages b/messages
index a5c1966..364fc91 100644
--- a/messages
+++ b/messages
@@ -1 +1,2 @@
 Hello, world
+Howdy all

编辑:

一个轻微的改进是使用git apply来代替patch:

git stash show -p | git apply --reverse

或者,你也可以使用git apply -R作为git apply——reverse的简写。

我最近发现这真的很方便……