我们在日常的开发中,需要将自己的改动给到其他同事时,经常需要将改动打补丁 (patch) 后进行处理。
git format-patch 是一种帮助开发人员从其 git 提交创建补丁的命令。这个命令很有用,可以用于各种原因,如代
码审查、在分支之间共享代码更改以及向邮件列表提交补丁。对于 git format-patch 的补丁,可以使用 git am 命
令进行打入。
本文我们将详细解释 git format-patch 和 git am 命令以及如何有效使用它。
$ git format-patch filename.patch
git format-patch 命令可以输出多种格式的补丁。默认情况下,它会以 Git 邮箱格式输出补丁。但是,您可以使用
以下选项来输出其他格式的补丁:
--stdout
选项将补丁输出到标准输出,而不是将其写入文件。当您想将补丁传输到其他 Git 命令或工具时,
此选项非常有用。
--mbox
选项以 mbox 格式输出补丁。当通过电子邮件发送补丁时,此格式非常有用。
--raw
选项以原始格式输出补丁。当您想向非 Git 存储库应用补丁时,此格式非常有用。
--numbered
选项按顺序编号补丁。当您想生成多个补丁并保持它们的特定顺序时,此选项非常有用。
git format-patch 命令可以通过指定提交范围或逐个指定一个或多个提交来使用。
format-patch可以基于分支进行打包,也可以基于上几次更新内容打包。
下面将演示其几中用法。
# 分支的提交记录
$ git log --oneline
b9709d6 (HEAD -> branch_a, origin/branch_a) branch_a | update a.txt | add new.txt
105370e add f.txt
46409d9 add e.txt
082352c add d.txt
d8f4266 add c.txt
a5e5961 add b.txt
4db21f4 add a.txt
# 打包最近的一个patch
$ git format-patch HEAD^
0001-branch_a-update-a.txt-add-new.txt.patch
# 打包最近的两个patch
$ git format-patch HEAD^^
0001-add-f.txt.patch
0002-branch_a-update-a.txt-add-new.txt.patch
# 打包最近的三个patch
$ git format-patch HEAD^^^
0001-add-e.txt.patch
0002-add-f.txt.patch
0003-branch_a-update-a.txt-add-new.txt.patch
# 有几个^就打包几个patch的内容或
# 可以使用git format-patch -n,两者是等价的
# 打包最近的一个patch
$ git format-patch -1
0001-branch_a-update-a.txt-add-new.txt.patch
# 打包最近的两个patch
$ git format-patch -2
0001-add-f.txt.patch
0002-branch_a-update-a.txt-add-new.txt.patch
# 打包版本n1与n2之间的patch
$ git format-patch -n1 -n2
# 可以打包版本2,3的patch
# 但是测试会把最近4个包都打包出来
$ git format-patch -1 -4
0001-add-d.txt.patch
0002-add-e.txt.patch
0003-add-f.txt.patch
0004-branch_a-update-a.txt-add-new.txt.patch
# 测试会把最近3个包都打包出来
$ git format-patch -1 -3
0001-add-e.txt.patch
0002-add-f.txt.patch
0003-branch_a-update-a.txt-add-new.txt.patch
# 测试会把最近2个包都打包出来
$ git format-patch -1 -2
0001-add-f.txt.patch
0002-branch_a-update-a.txt-add-new.txt.patch
# 测试会把最近1个包都打包出来
$ git format-patch -1 -1
0001-branch_a-update-a.txt-add-new.txt.patch
# 某次提交以后的所有patch
# 082352c add d.txt
# 不包含此次提交
$ git format-patch -s 082352c
# 等价
$ git format-patch 082352c
0001-add-e.txt.patch
0002-add-f.txt.patch
0003-branch_a-update-a.txt-add-new.txt.patch
# 某次提交之前的几次提交
# 包含本次提交
$ git format-patch -2 082352c
0001-add-c.txt.patch
0002-add-d.txt.patch
# 某两次提交之间的所有patch
# 不包含start的提交
# 46409d9 add e.txt
# 082352c add d.txt
# d8f4266 add c.txt
# a5e5961 add b.txt
$ git format-patch a5e5961..46409d9
0001-add-c.txt.patch
0002-add-d.txt.patch
0003-add-e.txt.patch
# 将所有patch输出到一个指定位置的指定文件
$ git format-patch commit --stdout > filename.patch
$ git format-patch d8f4266 --stdout > d8f4266.patch
# 经常使用的格式
# patch_name的格式类似于0001-bdbfc4ca3.patch
$ git format-patch -1 -k commit --stdout > dir_name/patch_name
$ git format-patch -1 -k d8f4266 --stdout> /tmp/0001-d8f4266.patch
# 当前分支所有超前branch_a的提交
$ git log --oneline
b1786d3 (HEAD -> branch_b, origin/branch_b) branch_b | update a.txt | add new.txt
b9709d6 (origin/branch_a) branch_a | update a.txt | add new.txt
105370e add f.txt
46409d9 add e.txt
082352c add d.txt
d8f4266 add c.txt
a5e5961 add b.txt
4db21f4 add a.txt
# 当前分支比较大
$ git format-patch -M origin/branch_a
0001-branch_b-update-a.txt-add-new.txt.patch
# 从origin到指定提交的所有patch
$ git format-patch -s --root origin
0001-add-a.txt.patch
0002-add-b.txt.patch
0003-add-c.txt.patch
0004-add-d.txt.patch
0005-add-e.txt.patch
0006-add-f.txt.patch
0007-branch_a-update-a.txt-add-new.txt.patch
0008-branch_b-update-a.txt-add-new.txt.patch
0009-branch_c-update-a.txt-delete-e.txt.patch
0010-develop-add-test.txt.patch
# 要为最后两个提交生成补丁
$ git format-patch HEAD~2..HEAD
0001-branch_a-update-a.txt-add-new.txt.patch
0002-branch_b-update-a.txt-add-new.txt.patch
# 当前分支和某个分支差异的patch
# 当前分支比价大
$ git format-patch branch_a
0001-branch_b-update-a.txt-add-new.txt.patch
在使用 git am 前,首先要使用 git am –abort,用来放弃以前的 am 信息,否则可能会遇到这样的错误:
.git/rebase-apply still exists but mbox given
基本语法:
# 打入patch
$ git am filename.patch
# 等价于
$ git am -k filename.patch
# 不应用修补程序,而是输出diffstat作为输入
$ git apply --stat filename.patch
# 不应用修补程序,而是查看修补程序是否适用
$ git apply --check filename.patch
git apply 命令通过修改现有文件或创建新文件来将补丁应用到代码库中。
$ git apply filename.patch
git apply 命令还可以与各种选项一起使用,例如 --check
以检查是否能干净地应用补丁,--reject
以创建带有
拒绝更改的补丁文件,以及 --index
以将更改添加到Git索引。
git am:
$ git am filename.patch
# 一次打入多个文件
$ git am *.patch
touch a.txt
git add a.txt
git commit -m "add a.txt"
touch b.txt
git add b.txt
git commit -m "add b.txt"
touch c.txt
git add c.txt
git commit -m "add c.txt"
touch d.txt
git add d.txt
git commit -m "add d.txt"
touch e.txt
git add e.txt
git commit -m "add e.txt"
touch f.txt
git add f.txt
git commit -m "add f.txt"
$ ls
a.txt b.txt c.txt d.txt e.txt f.txt
$ git log --oneline
3d47274 (HEAD -> master) add f.txt
f21670a add e.txt
3be00b5 add d.txt
8480613 add c.txt
054694f add b.txt
6cd2b56 add a.txt
$ git show 3d47274
commit 3d472749b02447f9e295bc924c95044a40a8ab42 (HEAD -> master)
Author: zsx242030 <2420309401@qq.com>
Date: Mon May 22 21:34:28 2023 +0800
add f.txt
diff --git a/f.txt b/f.txt
new file mode 100644
index 0000000..e69de29
$ git format-patch -1 -k 3d47274
0001-add-f.txt.patch
$ cat 0001-add-f.txt.patch
From 3d472749b02447f9e295bc924c95044a40a8ab42 Mon Sep 17 00:00:00 2001
From: zsx242030 <2420309401@qq.com>
Date: Mon, 22 May 2023 21:34:28 +0800
Subject: add f.txt
---
f.txt | 0
1 file changed, 0 insertions(+), 0 deletions(-)
create mode 100644 f.txt
diff --git a/f.txt b/f.txt
new file mode 100644
index 0000000..e69de29
--
2.14.1.windows.1
如果有额外信息需要补充,但又不想放在提交消息中说明,可以编辑这些补丁文件,在第一个 ---
行之前添加说
明,但不要修改下面的补丁正文。这样,其它开发者能阅读,但在采纳补丁时不会将此合并进来。
$ git reset --hard HEAD^
HEAD is now at f21670a add e.txt
$ git log --oneline
f21670a (HEAD -> master) add e.txt
3be00b5 add d.txt
8480613 add c.txt
054694f add b.txt
6cd2b56 add a.txt
$ git apply --stat 0001-add-f.txt.patch
f.txt | 0
1 file changed, 0 insertions(+), 0 deletions(-)
$ git apply --check 0001-add-f.txt.patch
$ git am 0001-add-f.txt.patch
Applying: add f.txt
$ git log --oneline
b8eafce (HEAD -> master) add f.txt
f21670a add e.txt
3be00b5 add d.txt
8480613 add c.txt
054694f add b.txt
6cd2b56 add a.txt
# git apply执行之后需要自己提交文件
$ git apply 0001-add-f.txt.patch
$ ls
a.txt b.txt c.txt d.txt e.txt f.txt
$ git log --oneline
b8eafce (HEAD -> master) add f.txt
f21670a add e.txt
3be00b5 add d.txt
8480613 add c.txt
054694f add b.txt
6cd2b56 add a.txt
$ git status
On branch master
Untracked files:
(use "git add ..." to include in what will be committed)
0001-add-f.txt.patch
f.txt
nothing added to commit but untracked files present (use "git add" to track)
$ git add f.txt
$ git commit -m "add f.txt"
[master de26d5b] add f.txt
1 file changed, 0 insertions(+), 0 deletions(-)
create mode 100644 f.txt
$ git log --oneline
de26d5b (HEAD -> master) add f.txt
f21670a add e.txt
3be00b5 add d.txt
8480613 add c.txt
054694f add b.txt
6cd2b56 add a.txt
如果只是想生效改动而不需要直接提交代码,可以使用 git apply 代替 git am。
# master
touch a.txt
git add a.txt
git commit -m "add a.txt"
touch b.txt
git add b.txt
git commit -m "add b.txt"
$ git log --oneline
4ac5e82 (HEAD -> master) add b.txt
ce93e86 add a.txt
# featurea
$ cat >> a.txt << EOF
11
aa
22
bb
33
cc
EOF
git add .
git commit -m "update a.txt"
$ git log --oneline
11ea22f (HEAD -> featurea) update a.txt
4ac5e82 (master) add b.txt
ce93e86 add a.txt
# featureb
$ cat >> a.txt << EOF
aa
11
bb
22
cc
33
EOF
git add a.txt
git commit -m "update a.txt"
$ git log --oneline
6c9e62d (HEAD -> featureb) update a.txt
4ac5e82 (master) add b.txt
ce93e86 add a.txt
featurea 分支和 featureb 分支的内容是有冲突的,featurea 先开发完了,然后 featureb 又进行了修改,会导致
a.txt 文件冲突。
# featureb
$ git format-patch -1
0001-update-a.txt.patch
# featurea
$ git am 0001-update-a.txt.patch
error: patch failed: a.txt:0
error: a.txt: patch does not apply
hint: Use 'git am --show-current-patch=diff' to see the failed patch
Applying: update a.txt
Patch failed at 0001 update a.txt
When you have resolved this problem, run "git am --continue".
If you prefer to skip this patch, run "git am --skip" instead.
To restore the original branch and stop patching, run "git am --abort".
由于两个分支都同时操作了某个文件,导致冲突,这种情况下可以使用下面的方法解决冲突:
1、执行命令 git apply --reject finame.patch 自动合入 patch 中不冲突的代码改动,同时保留冲突的部分。这些
存在冲突的改动内容会被单独存储到目标源文件的相应目录下,以后缀为 .rej 的文件进行保存。
$ git apply --reject 0001-update-a.txt.patch
Checking patch a.txt...
error: while searching for:
error: patch failed: a.txt:0
Applying patch a.txt with 1 reject...
Rejected hunk #1.
会生成一个 a.txt.rej 文件:
$ cat a.txt.rej
diff a/a.txt b/a.txt (rejected hunks)
@@ -0,0 +1,6 @@
+aa
+11
+bb
+22
+cc
+33
2、依据上面生成的 *.rej 文件内容逐个手动解决冲突,然后删除这些 *.rej 文件。
我们解决 a.txt 文件中的冲突,将内容替换为两者的合并:
11
aa
aa
11
22
bb
bb
22
33
cc
cc
33
然后删除 a.txt.rej 文件:
$ rm -rf a.txt.rej
完成这一步骤的操作后,我们就可以继续执行 git am 的过程了。
$ git am 0001-update-a.txt.patch
fatal: previous rebase directory .git/rebase-apply still exists but mbox given.
# 如果报上面的错误
# 解决方法
$ git am --abort
# 继续执行
$ git am 0001-update-a.txt.patch
error: a.txt: does not match index
hint: Use 'git am --show-current-patch=diff' to see the failed patch
Applying: update a.txt
Patch failed at 0001 update a.txt
When you have resolved this problem, run "git am --continue".
If you prefer to skip this patch, run "git am --skip" instead.
To restore the original branch and stop patching, run "git am --abort".
3、执行完上述命令之后执行 git status 查看当前改动过的以及新增的文件,确保没有多添加或少添加文件。
$ git status
On branch featurea
You are in the middle of an am session.
(fix conflicts and then run "git am --continue")
(use "git am --skip" to skip this patch)
(use "git am --abort" to restore the original branch)
Changes not staged for commit:
(use "git add ..." to update what will be committed)
(use "git restore ..." to discard changes in working directory)
modified: a.txt
4、执行命令 git add 添加所有有改动的文件,将文件都添加到暂存区。
$ git add a.txt
5、执行命令 git am --continue 或者是 git am --resolved 继续被中断的 patch 合入操作。合入完成后,会有提示
信息输出。
$ git am --continue
Applying: update a.txt
6、执行命令 git log 确认合入状态。
$ git log --oneline
1add6d0 (HEAD -> featurea) update a.txt
11ea22f update a.txt
4ac5e82 (master) add b.txt
ce93e86 add a.txt
7、至此,带有冲突代码的 patch 合入就操作完成了。如果要修改 commit 信息,执行 git commit --amend 命令
即可。
$ git commit --amend -m "second update a.txt"
$ git log --oneline
3c27164 (HEAD -> featurea) second update a.txt
11ea22f update a.txt
4ac5e82 (master) add b.txt
ce93e86 add a.txt
如果想让 Git 更智能地处理冲突,可以用 -3 选项进行三方合并。如果当前分支未包含该补丁的基础代码或其祖
先,那么三方合并就会失败,所以该选项默认为关闭状态。一般来说,如果该补丁是基于某个公开的提交制作而成
的话,你总是可以通过同步来获取这个共同祖先,所以用三方合并选项可以解决很多麻烦。
还是以上面的冲突为例进行演示:
# featurea
$ git am -3 0001-update-a.txt.patch
Applying: update a.txt
error: Failed to merge in the changes.
Using index info to reconstruct a base tree...
M a.txt
Falling back to patching base and 3-way merge...
Auto-merging a.txt
CONFLICT (content): Merge conflict in a.txt
Patch failed at 0001 update a.txt
The copy of the patch that failed is found in: .git/rebase-apply/patch
When you have resolved this problem, run "git am --continue".
If you prefer to skip this patch, run "git am --skip" instead.
To restore the original branch and stop patching, run "git am --abort".
$ cat a.txt
<<<<<<< HEAD
11
aa
22
bb
33
cc
=======
aa
11
bb
22
cc
33
>>>>>>> update a.txt
# 解决冲突
$ cat a.txt
11
aa
aa
11
22
bb
bb
22
33
cc
cc
33
$ git add a.txt
$ git am --continue
Applying: update a.txt
$ git commit --amend -m "second update a.txt"
$ git log --oneline
bf7a56b (HEAD -> featurea) second update a.txt
b7622ab update a.txt
2dc23a3 (master) add b.txt
b568674 add a.txt
对于一次应用多个补丁时所用的 mbox 格式文件,可以用 am 命令的交互模式选项 -i,这样就会在打每个补丁前
停住,询问该如何操作:
$ git am -3 -i 0001-update-a.txt.patch
Commit Body is:
--------------------------
update a.txt
--------------------------
Apply? [y]es/[n]o/[e]dit/[v]iew patch/[a]ccept all: v
---
a.txt | 6 ++++++
1 file changed, 6 insertions(+)
diff --git a/a.txt b/a.txt
index e69de29..c1ba934 100644
--- a/a.txt
+++ b/a.txt
@@ -0,0 +1,6 @@
+aa
+11
+bb
+22
+cc
+33
--
2.14.1.windows.1
Commit Body is:
--------------------------
update a.txt
--------------------------
Apply? [y]es/[n]o/[e]dit/[v]iew patch/[a]ccept all: n
在多个补丁要打的情况下,这是个非常好的办法,一方面可以预览下补丁内容,同时也可以有选择性的接纳或跳过
某些补丁。