作为程序员,了解diff&patch命 令是非常必要的。比如说我们发现某个项目有bug代码,而自己又没有svn的提交权限,那么此时最合适的解决方法就是用diff命令做一个补丁发给项目成 员。项目成员通过patch命令可以立刻知道你的意图。有人会说直接传一个新文件不是更简单?不要忘了,一个patch文件尺寸更小传输更快,而且可以明 显的看到都做了哪些修改。
保证当前目录是demo名录:
# mkdir demo
# cd demo
先模拟一个项目目录old:
# mkdir -p old/a/b
# vi old/a/b/foo.txt
old_line_1
old_line_2
假设我们发现项目old有bug代码,下面我们先拷贝一个新目录new,并在此修改bug代码:
# cp -r old new
# vi new/a/b/foo.txt
new_line_1
new_line_2
保证old和new两个目录都在当前目录下,下面就可以使用diff命令了,不要使用绝对路径,而应该使用相对路径,至于原因,看到文章结尾你就清楚了:
# LC_ALL=C TZ=UTC0 diff -Naur old new > foo.patch
如果不在意字符集,时差等问题,也可以省略LC_ALL=C TZ=UTC0环境变量:
# diff -Naur old new > foo.patch
其中-Naur参数属于固定打法,不管是对一个文件,还是对一个目录,在使用这个参数基本就可以了。
大概浏览一下补丁文件:
# cat foo.patch
diff -Naur old/a/b/foo.txt new/a/b/foo.txt
--- old/a/b/foo.txt 2009-12-07 20:40:07.000000000 +0800
+++ new/a/b/foo.txt 2009-12-07 20:41:51.000000000 +0800
@@ -1,2 +1,2 @@
-old_line_1
-old_line_2
+new_line_1
+new_line_2
加减号后面的内容是有用的内容,其他的内容是方便你查阅的相关信息内容,补丁制作完成。
此时的文件目录结构大概如下所示:
#tree
demo
|-- old
| `-- a
| `-- b
| `-- foo.txt
|-- new
| `-- a
| `-- b
| `-- foo.txt
`-- foo.patch
下面看看如何使用patch来应用补丁,要注意的是当前目录是demo,试试下面命令:
# patch -p0 < foo.patch
patching file old/a/b/foo.txt
这里唯一需要说明的是p0的含义,因为在foo.patch补丁文件里的路径信息是这样的:
--- old/a/b/foo.txt
p表示跳过几级目录,因为是在demo目录下使用的patch命令,old目录就在demo目录下,所以不必跳过任何目录,而应该使用old/a/b/foo.txt完整路径,所以此时使用的是p0。
查看一下目标文件,你会发现内容已经修改成新的了:
# cat old/a/b/foo.txt
new_line_1
new_line_2
此时如果你再次使用patch命令,系统会问你是否想还原:
# patch -p0 < foo.patch
patching file old/a/b/foo.txt
Reversed (or previously applied) patch detected! Assume -R? [n] y
查看一下目标文件,你会发现内容已经还原成旧的了:
# cat old/a/b/foo.txt
old_line_1
old_line_2
保证当前目录是demo名录:
# mkdir demo
# cd demo
先模拟一个项目目录old:
# mkdir -p old/a/b
# vi old/a/b/foo.txt
old_line_1
old_line_2
假设我们发现项目old有bug代码,下面我们先拷贝一个新目录new,并在此修改bug代码:
# cp -r old new
# vi new/a/b/foo.txt
new_line_1
new_line_2
保证old和new两个目录都在当前目录下,下面就可以使用diff命令了,不要使用绝对路径,而应该使用相对路径,至于原因,看到文章结尾你就清楚了:
# LC_ALL=C TZ=UTC0 diff -Naur old new > foo.patch
如果不在意字符集,时差等问题,也可以省略LC_ALL=C TZ=UTC0环境变量:
# diff -Naur old new > foo.patch
其中-Naur参数属于固定打法,不管是对一个文件,还是对一个目录,在使用这个参数基本就可以了。
大概浏览一下补丁文件:
# cat foo.patch
diff -Naur old/a/b/foo.txt new/a/b/foo.txt
--- old/a/b/foo.txt 2009-12-07 20:40:07.000000000 +0800
+++ new/a/b/foo.txt 2009-12-07 20:41:51.000000000 +0800
@@ -1,2 +1,2 @@
-old_line_1
-old_line_2
+new_line_1
+new_line_2
加减号后面的内容是有用的内容,其他的内容是方便你查阅的相关信息内容,补丁制作完成。
此时的文件目录结构大概如下所示:
#tree
demo
|-- old
| `-- a
| `-- b
| `-- foo.txt
|-- new
| `-- a
| `-- b
| `-- foo.txt
`-- foo.patch
下面看看如何使用patch来应用补丁,要注意的是当前目录是demo,试试下面命令:
# patch -p0 < foo.patch
patching file old/a/b/foo.txt
这里唯一需要说明的是p0的含义,因为在foo.patch补丁文件里的路径信息是这样的:
--- old/a/b/foo.txt
p表示跳过几级目录,因为是在demo目录下使用的patch命令,old目录就在demo目录下,所以不必跳过任何目录,而应该使用old/a/b/foo.txt完整路径,所以此时使用的是p0。
查看一下目标文件,你会发现内容已经修改成新的了:
# cat old/a/b/foo.txt
new_line_1
new_line_2
此时如果你再次使用patch命令,系统会问你是否想还原:
# patch -p0 < foo.patch
patching file old/a/b/foo.txt
Reversed (or previously applied) patch detected! Assume -R? [n] y
查看一下目标文件,你会发现内容已经还原成旧的了:
# cat old/a/b/foo.txt
old_line_1
old_line_2