在大型项目(如 Chromium 这类超大型仓库)的日常开发中,我们经常会遇到这样的场景:
在一个临时分支上解决了大量 merge 冲突,但工作区里还混杂着其他不相关的变更。现在需要把特定几个目录的解决结果,精准地迁移到另一个目标分支上。
这时候,git diff + git apply 的组合就是最优雅的解决方案。
背景:为什么不直接 cherry-pick 或 merge?
- cherry-pick 依赖 commit 历史的匹配,如果两个分支的基础不同,很容易产生大量新冲突。
- merge 会把所有变更都带过去,无法精准筛选。
- patch 方式 可以只导出你关心的目录,干净、精准、可控。
第一步:只针对指定目录生成 diff
git diff 支持在命令末尾用 -- 指定路径,可以精准过滤出你需要的目录:
# 对比目标分支,只导出 chrome/app/ 和 extensions/ 两个目录的变更
git diff origin/target-branch -- chrome/app/ extensions/ > /tmp/chrome_app_ext.patch
如果变更已经 git add 进了暂存区(staged),用 --cached:
git diff --cached -- chrome/app/ extensions/ > /tmp/chrome_app_ext.patch
如果想对比某个具体的 commit:
git diff <base-commit>..HEAD -- chrome/app/ extensions/ > /tmp/chrome_app_ext.patch
第二步:在目标分支上应用 patch
切换到目标分支,先用 --check 做一次"预检",确认 patch 能否干净应用:
git checkout target-branch
# 预检,不实际修改文件
git apply --check /tmp/chrome_app_ext.patch
如果预检通过,正式应用:
git apply /tmp/chrome_app_ext.patch
第三步:遇到冲突怎么办?
如果目标分支和 patch 有冲突,直接 git apply 会报错退出。这时候加上 --3way 参数,Git 会自动产生冲突标记(类似 merge 冲突),让你手动解决:
git apply --3way /tmp/chrome_app_ext.patch
解决完冲突后,正常 git add + git commit 即可。
完整流程示例
# === 在 merge 分支上 ===
# 只导出 chrome/app/ 和 extensions/ 的变更
git diff origin/target-branch -- chrome/app/ extensions/ > /tmp/chrome_app_ext.patch
# === 切换到目标分支 ===
git checkout target-branch
# 预检
git apply --check /tmp/chrome_app_ext.patch
# 应用(如果有冲突用 --3way)
git apply --3way /tmp/chrome_app_ext.patch
# 提交
git add .
git commit -m "apply resolved changes from chrome/app and extensions"
小结
| 场景 | 推荐命令 |
|---|---|
| 只导出指定目录的变更 | git diff <ref> -- path1/ path2/ > my.patch |
| 干净应用 patch | git apply my.patch |
| 有冲突时应用 | git apply --3way my.patch |
| 预检 patch 是否可用 | git apply --check my.patch |
这个方法特别适合 Chromium 这类超大型仓库的升核(内核版本升级)场景:在临时分支上完成 AI 辅助冲突解决后,只把属于自己业务的目录变更精准地 patch 到正式分支,避免引入无关的噪音变更。