path 修补文件命令
path 修补文件命令
功能:修补文件,使用补丁文件,对源文件进行更改。更改方式根据参数设定。并且能够使用命令配合修补文件的方式一次性修补大批文件。
使用语法:
patch [option] [origfile] [patchfile]
参数选项:
输入选项 | – | – |
-p NUM | –strip=NUM | 去除相对路径层次的数目 |
-F LINES | –fuzz LINES | 设置鉴别列数 |
-I | –ignore–whitespace | 忽略修补数据与输入数据的跳格,空格字符 |
-c | –context | 把修补数据解释成关联性的差异 |
-e | –ed | 把修补数据解释成 ed 指令可用的叙述文件 |
-n | –normal | 把修补数据解释成一般性的差异 |
-u | –unified | 把修补数据解释成一致化的差异 |
-N | –forward | 忽略修补的数据较原始文件的版本更旧,或该奔波的修补数据已使用过 |
-R | –reverse | 假设修补数据是由新旧文件交换位置而产生 |
-i PATCHFILE | –input=PATCHFILE | 读取指定的修补文件 |
输出选项 | – | – |
-o FILE | –output=FILE | 设置输出文件的名称,修补过的文件会以该名称存放 |
-r FILE | –reject-file=FILE | Output rejects to FILE |
-D NAME | –ifdef=NAME | 用指定的符号把改变的地方标示出来 |
-m | –merge | Merge using conflict markers instead of creating reject files |
-E | –remove-empty-files | 若修补过后输出的文件其内容是一片空白,则移除该文件 |
-Z | –set-utc | 把修补过的文件更改,存取时间设为UTC |
-T | –set-time | 此参数的效果和指定"-Z"参数类似,但以本地时间为主 |
– | –quoting-style=WORD | 使用WORD引述类型显示项目名称,可设定值有literal,shell,shell-always,c,escape |
备份和版本控制选项 | – | – |
-b | –backup | 备份每一个原始文件 |
–backup-if-mismatch | 在修补数据不完全吻合,且没有刻意指定要备份文件时,才备份文件 | |
–no-backup-if-mismatch | 在修补数据不完全吻合,且没有刻意指定要备份文件时,不要备份文件 | |
-V STYLE | –version-control=STYLE | 用"-b"参数备份目标文件后,备份文件的字尾会被加上一个备份字符串,这个字符串不仅可用"-z"参数变更, |
当使用"-V"参数指定不同备份方式时,也会产生不同字尾的备份字符串 | ||
-B PREFIX | –prefix=PREFIX | 设置文件备份时,附加在文件名称前面的字首字符串,该字符串可以是路径名称 |
-Y PREFIX | –basename-prefix=PREFIX | 设置文件备份时,附加在文件基本名称开头的字首字符串 |
-z SUFFIX | –suffix=SUFFIX | 此参数的效果和指定"-B"参数类似,差别在于修补作业使用的路径与文件名若为src/linux/fs/super.c,加上"backup/"字符串后,文件super.c会备份于/src/linux/fs/backup目录里 |
-g NUM | –get=NUM | 设置以RSC或SCCS控制修补作业 |
其他选项 | – | – |
-t | –batch | 自动略过错误,不询问任何问题 |
-f | –force | 此参数的效果和指定"-t"参数类似,但会假设修补数据的版本为新版本 |
-s | –quiet 或 --silent | 不显示指令执行过程,除非发生错误 |
–verbose | 显示详细的过程信息 | |
–dry-run | 实际上不改变任何文件;演示讲会发生什么 | |
–posix | 符合POSIX标准 | |
-d DIR | –directory=DIR | 先改变工作目录到指定的目录 |
–reject-format=FORMAT | Create ‘context’ or ‘unified’ rejects | |
–binary | 以二进制方式读写数据 | |
–read-only=BEHAVIOR | 如何处理只读输入文件:“忽视”,他们是只读的,“警告”(默认),或“失败” |
注意:
patch 命令(默认)使用从标准输入读入的源文件 < PATCHFILE ,但是使用 -i PATCHFILE 设置。源文件包含由 diff 命令产生的差别列表(或者 diff 列表)。差异列表是比较两个文件和构建关于如何纠正差别的指示信息的结果。差异列表有三种格式:正常、上下文或者是 ed 编辑器风格。patch 命令确定差异列表格式,除非被 -c、-e 或 -n 标志否决。默认,ORIGFILE 被PATCHFILE 替换。若ORIGFILE(原始文件)不存在时,PATCHFILE(补丁文件)根据差别列表,创建 ORIGFILE 文件。指定 -b 标志时,ORIGFILE(原始文件)会备份在自身的文件中,只是在文件名后附加了后缀 .orig。使用 -o 标志也可以指定输出的目的地。
patch常用选项:
- -r 是一个递归选项,设置了这个选项,diff会将两个不同版本源代码目录中的所有对应文件全部都进行一次比较,包括子目录文件。
- -N 选项确保补丁文件将正确地处理已经创建或删除文件的情况。
- -u 选项以统一格式创建补丁文件,这种格式比缺省格式更紧凑些。
- -p0 选项从当前目录查找目的文件(夹)(直接使用补丁文件里面指定的路径)
- -p1 选项忽略掉第一层目录,从当前目录查找(去掉补丁文件指定路径最左的第1个’/'及前面所有内容)。
- -E 选项说明如果发现了空文件,那么就删除它
- -R 选项说明在补丁文件中的“新”文件和“旧”文件现在要调换过来了(实际上就是给新版本打补丁,让它变成老版本)
操作练习:
具体操作命令如下:
针对单个文件的修改补丁
$ diff –uN from_file to_file > demo_to_file.patch
$ patch –p0 < demo_to_file.patch # 对 from 文件操作,使其提升为 to 文件
$ patch –RE –p0 < demo_to_file.patch # 对当前文件操作,使其恢复为 from 文件
针对文件目录下所有文件的修改补丁
$ diff –uNr from_dir to_dir > demo_to_dir.patch
$ patch –p1 < demo_to_dir.patch # 打补丁,使得成为新版本文件
$ patch –R –p1 < demo_to_dir.patch # 恢复补丁,使其回到补丁前版本
创建如下示例工程目录结构:
其中主要文件内容如下:
- Makefile
BOOT_DIR=$(shell pwd)
CC=gcc
obj_src=${shell find $(BOOT_DIR) -name "*.c"}
objs=${patsubst %.c,%.o,$(obj_src)}
deps=${patsubst %.c,%.d,$(obj_src)}
CFLAGS=-I${BOOT_DIR}/common
OBJ=demo
all:${objs}
${CC} ${CFLAGS} $< -o ${OBJ}
%.o:%.c
${CC} ${CFLAGS} $< -c -o $@
clean:
-rm ${objs}
-rm ${OBJ}
- common/base.h
#ifndef BASE__H
#define BASE__h
#define VERSION 1
#endif
- ext/ext.t
# 空文件
- main/main.c
#include <stdio.h>
#include "base.h"
int main()
{
printf("Hello world!\n");
printf("Current Version : %d\n", VERSION);
return 0;
}
可以尝试如下操作, 我们将通过修改其中 common/base.h
文件中的版本信息宏定义来进行演示,先 make
构建工程进行演示,如下:
可以看到其中的当前显示版本号为 1 ,接下来将进行修改。复制一个当前工程为 new_src
,
$ cp -rf old_src new_src
$ cd new_src/common/
$ vim base.h
将对应的版本号修改为 2
或者其他。具体如下:
生成补丁操作:
$ diff -uNr old_src new_src > demo.patch
生成补丁内容如下:
这里生成的补丁文件 demo.patch
就可以用来将 old_src
与 new_src
进行同步了。当我们只有 old_src
时,我们拿到 demo.patch
补丁,通过如下操作可以对源代码进行修改同步,而不需要逐行修改,具体如下:
$ mv demo.patch old_src/
$ cd old_src
$ patch -p1 < demo.patch
显示如下:
可以看到 patch
的输出结果,修改了对应的文件 common/base.h
,并且经过编译后程序输出的版本信息已经修改为新版本的 2
此外,还能对新版本进行恢复操作,具体如下:
$ mv demo.patch new_src/
$ cd new_src
$ patch -R -p1 < demo.patch
显示如下:
补充:svn生成补丁文件和打补丁文件
生成补丁文件:
svn diff > patchFile 整个工程的变动生成补丁
或 svn diff file > patchFile 某个文件单独变动的补丁
svn回滚:
svn revert FILE 单个文件回滚
svn revert DIR --depth=infinity 整个目录进行递归回滚
打patch:
patch -p0 < test.patch -p0 选项要从当前目录查找目的文件
patch -p1 < test.patch -p1 选项要从当前目录查找目的文件,不包含patch中的最上级目录
例如两个版本以a,b开头,而a,b并不是真正有效地代码路径,则这时候需要使用"-p1"参数。
a/src/…
b/src/…
总结
从 linux
的 patch
命令的学习中,我们发现原来在 linux
环境下修改保存文件的版本信息如此简单,其实在很多的代码版本管理中也都是用到了这样的操作和版本变更的信息记录,例如经常用到的 git
svn
等等工具,我们都能够在其中发现 patch
的声影。
通过本章的学习,相信大家在日后码代码的过程中,再也不会遇到类似 patch
文件后还采用逐行修改的方式进行了。 ????