背景

当我们修改了代码准备提交时,本地的改动可能包含了不能提交的调试语句,还可能需要拆分成多个细粒度的 pactch

本文将介绍如何使用 git add -p 来交互式选择代码片段,辅助整理出所需的 patch

官方介绍

先贴个帮助信息供参考

英文版本:

  1. -p, --patch
  2. Interactively choose hunks of patch between the index and the work tree and add them to
  3. the index. This gives the user a chance to review the difference before adding modified
  4. contents to the index.
  5. This effectively runs add --interactive, but bypasses the initial command menu and
  6. directly jumps to the patch subcommand. See Interactive mode for details.

中文版本:

  1. -p, --patch
  2. 交互地在索引和工作树之间选择补丁块并将它们添加到索引中。这让用户有机会在将修改后的内容添加到索引之前查看差异。
  3. 这可以有效地运行 add --interactive,但是会绕过初始命令菜单,而直接跳转到 patch 子命令。有关详细信息,请参见`‘交互模式’'。

demo 视频版

以下文字版例子对应的视频演示

demo 文字版

我们造个例子来说明,假设我们本次完成了两个功能,fun1 和 fun2,希望分开提交。另外在修改过程中还引入了一些调试的打印,是不需要提交的。

代码的 diff 如下

  1. --git a/demo.c b/demo.c
  2. index 0473c1a..76cfb22 100644
  3. --- a/demo.c
  4. +++ b/demo.c
  5. @@ -1,16 +1,31 @@
  6. #include <stdio.h>
  7. +void fun1()
  8. +{
  9. + printf("before hello world\n");
  10. +}
  11. +
  12. void demo()
  13. {
  14. ;
  15. }
  16. +void fun2()
  17. +{
  18. + printf("after hello world\n");
  19. +}
  20. +
  21. int main()
  22. {
  23. + fun1();
  24. printf("hello world\n");
  25. + printf("debug %s %d\n", __func__, __LINE__);
  26. printf("hello world\n");
  27. printf("hello world\n");
  28. printf("hello world\n");
  29. + printf("debug %s %d\n", __func__, __LINE__);
  30. printf("hello world\n");
  31. + fun2();
  32. demo();
  33. + printf("debug %s %d\n", __func__, __LINE__);
  34. }

此时直接 git add 会把整个文件的改动都加进来,不符合需求。

这正是 patch mode 发挥作用的地方,我们可以挑选一部分改动进行提交。

输入 git add -p 进入 patch mode , 此时 git 会自动将改动切分成多个片段,并展示第一个片段,提示你进行选择。

提示语句是 Stage this hunk [y,n,q,a,d,/,s,e,?]?

这些字母都是什么意思呢? 输入?回车,可以查看详细的帮助信息。

英文版本:

  1. y - stage this hunk
  2. n - do not stage this hunk
  3. q - quit; do not stage this hunk or any of the remaining ones
  4. a - stage this hunk and all later hunks in the file
  5. d - do not stage this hunk or any of the later hunks in the file
  6. g - select a hunk to go to
  7. / - search for a hunk matching the given regex
  8. j - leave this hunk undecided, see next undecided hunk
  9. J - leave this hunk undecided, see next hunk
  10. k - leave this hunk undecided, see previous undecided hunk
  11. K - leave this hunk undecided, see previous hunk
  12. s - split the current hunk into smaller hunks
  13. e - manually edit the current hunk
  14. ? - print help

中文版本:

  1. y - 暂存此区块
  2. n - 不暂存此区块
  3. q - 退出;不暂存包括此块在内的剩余的区块
  4. a - 暂存此块与此文件后面所有的区块
  5. d - 不暂存此块与此文件后面所有的 区块
  6. g - 选择并跳转至一个区块
  7. / - 搜索与给定正则表达示匹配的区块
  8. j - 暂不决定,转至下一个未决定的区块
  9. J - 暂不决定,转至一个区块
  10. k - 暂不决定,转至上一个未决定的区块
  11. K - 暂不决定,转至上一个区块
  12. s - 将当前的区块分割成多个较小的区块
  13. e - 手动编辑当前的区块
  14. ? - 输出帮助

对于我们的例子,git第一次自动给出的hunk很大,可以先执行 s 分割下。分割后第一个区块就只包含增加的 fun1 函数了。

  1. /* 太占篇幅,此处省略原始 hunk */
  2. Stage this hunk [y,n,q,a,d,/,s,e,?]? s /* 询问我们对第一个片段的处理,我们觉得太大,按 s 要求分割 */
  3. Split into 7 hunks. /* 可以看到,s 让 git 将原始片段进一步切分成了 7 个片段,接下来就是自动展示第一个片段 */
  4. @@ -1,7 +1,12 @@
  5. #include <stdio.h>
  6. +void fun1()
  7. +{
  8. + printf("before hello world\n");
  9. +}
  10. +
  11. void demo()
  12. {
  13. ;
  14. }
  15. Stage this hunk [y,n,q,a,d,/,j,J,g,e,?]? /* 询问我们对第一个片段的处理 */

输入 y 回车选中这个 fun1 的改动,git 就会自动展示下一个片段,继续询问我们。

这样对片段使用 yn,我们就可以只挑选出涉及 fun1 的改动,当我们确认后续没有 fun1 相关的改动时,就可以按 q 退出挑选了。

此时 git status 可以看到部分改动在暂存区中。

  1. $ git status
  2. On branch master
  3. Changes to be committed:
  4. (use "git reset HEAD <file>..." to unstage)
  5. modified: demo.c
  6. Changes not staged for commit:
  7. (use "git add <file>..." to update what will be committed)
  8. (use "git checkout -- <file>..." to discard changes in working directory)
  9. modified: demo.c

使用 git diff --cached 可以具体确认要提交的内容,是否符合我们的预期,只包含 fun1 的改动,不包含 fun2 和调试语句。

  1. $ git diff --cached
  2. diff --git a/demo.c b/demo.c
  3. index 0473c1a..b9fd4d4 100644
  4. --- a/demo.c
  5. +++ b/demo.c
  6. @@ -1,5 +1,10 @@
  7. #include <stdio.h>
  8. +void fun1()
  9. +{
  10. + printf("before hello world\n");
  11. +}
  12. +
  13. void demo()
  14. {
  15. ;
  16. @@ -7,6 +12,7 @@ void demo()
  17. int main()
  18. {
  19. + fun1();
  20. printf("hello world\n");
  21. printf("hello world\n");
  22. printf("hello world\n");

确认无误就可以提交第一个patch, 即 fun1 的改动了。

  1. git commit -m "fun1"

接下来继续使用 git add -p,配合s,y,'n'就可以进一步挑选出fun2的改动了。

如果要挑选的改动比较明确,可以直接使用 /来搜索到目标hunk,省去逐个片段判断的麻烦。例如执行 /fun2 来搜索包含 fun2 的代码片段。

git add -p 挑选完之后,建议使用 git diff --cached 确认下,或者在提交之后 git show 确认下改动,如有错漏,及时修正,多退少补。

大部分情况使用s y n就足够了。但如果有些改动是混合在一起的,无法使用s来分割,那就得用 e 来手工编辑了,下回分解吧。

blog: https://www.cnblogs.com/zqb-all/p/13020293.html

公众号:https://sourl.cn/MDcrJA

使用 git add -p 整理 patch的更多相关文章

  1. [Git] Use git add --patch for better commit history and mitigating bugs

    Let's split our changes into separate commits. We'll be able to check over our changes before stagin ...

  2. git add

    一.前言git add命令主要用于把我们要提交的文件的信息添加到索引库中.当我们使用git commit时,git将依据索引库中的内容来进行文件的提交. 二.基本git add <path> ...

  3. Git add 常见用法

        Git add   git add [参数]  [--] <路径> //作用就是将我们需要提交的代码从工作区添加到暂存区,就是告诉git系统,我们要提交哪些文件,之后就可以使用gi ...

  4. Git 常用命令整理

    初始化配置 #配置使用git仓库的人员姓名 git config --global user.name "Your Name Comes Here" #配置使用git仓库的人员em ...

  5. git add 命令详解

    或"表示git会例出索引库中的文件列表中的第个文件."-"表示git会例出索引库中的文件列表中的第个文件到第个文件.回车将执行.如果我们不输入任何东西,直接回车,将结束r ...

  6. git add --all 为啥不能添加空文件夹,这样设计的初衷是

    git add --all 为啥不能添加空文件夹,这样设计的初衷是? 好多项目还得弄个假文件在空文件夹里面占位 这个算设计失误吧,见 https://git.wiki.kernel.org/index ...

  7. (小组)Git 常用命令整理

    Git 常用命令整理 取得Git仓库 初始化一个版本仓库 git init Clone远程版本库 git clone git@xbc.me:wordpress.git 添加远程版本库origin,语法 ...

  8. git add 文档

    GIT-ADD(1) Git Manual GIT-ADD(1) NAME git-add - Add file contents to the index SYNOPSIS git add [-n] ...

  9. Git常见命令整理

    Git常见命令整理 + 注释 git init # 初始化本地git仓库(创建新仓库) git config --global user.name "xxx" # 配置用户名 gi ...

随机推荐

  1. redis系列之5----redis实战(redis与spring整合,分布式锁实现)

    本文是redis学习系列的第五篇,点击下面链接可回看系列文章 <redis简介以及linux上的安装> <详细讲解redis数据结构(内存模型)以及常用命令> <redi ...

  2. CSS的基础使用

    一,css是什么? CSS全称为“层叠样式表” ,与HTML相辅相成,实现网页的排版布局与样式美化 二,CSS使用方式 1.行内样式/内联样式(单一页面中使用) 借助于style标签属性,为当前的标签 ...

  3. LeetCode--LinkedList--203. Remove Linked List Elements(Easy)

    203. Remove Linked List Elements(Easy) 题目地址https://leetcode.com/problems/remove-linked-list-elements ...

  4. C# 判断文件格式的一些总结

    前提概述: 项目中 经常会有上传图片的地方  有的时候需要对图片类型做一些要求   这个时候就需要一些判断   虽然前段上传的时候可以去做类型的限制  或者后台接受的时候从file的type 中获取图 ...

  5. 内网穿透访问Vue项目的时候出现Invalid Host header解决办法

    适用场景: 在本地的Vue-cli3项目, 需要其他人浏览. 如果没有外网的服务器, 可以把自己的电脑当做服务器. 这时候需要外网的人能访问到自己的电脑. Mac内网穿透工具:natapp Inval ...

  6. 【跟我一起读 linux 源码 01】boot

    计算机启动流程在我的上一个学习计划<自制操作系统>系列中,已经从完全不知道,过渡到了现在的了如指掌了,虽然有些夸张,但整个大体流程已经像过电影一样在我脑海里了,所以在看 linux 源码的 ...

  7. An SWT error has occurred

    对话框标题:Problem Occurred 对话框内容:Unhandled event loop exception No more handles 对话框按钮:第一个超链接:Show Error ...

  8. 如何使用 Shebang Line (Python 虚拟环境)

    本文记录,如何在 Python Script 中使用 Shebang 行. Shebang Line 是什么: 也被叫做 Hashbang Line,只要是一个由,井号和叹号#!开头,并构成的字符序列 ...

  9. jstree 反选,测试400条数据左右有点卡

    $("#reversecheckallmachines").on("change", function () { var checkedNodes = []; ...

  10. 分布式项目配置工程,在项目间互通要先在linux下开启zookeeper

    一.编写配置文件 <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http: ...