`
zhengdl126
  • 浏览: 2509459 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类

用diff比较目录文件差异和patch备份源代码

 
阅读更多

 

=============??????? 新增加的文件无法更新

 

 

The next patch would create the file sc_patch/update20120730.sql,
which already exists!  Skipping patch.
1 out of 1 hunk ignored

 

===============

 

 

old:/var/www/ssc/
new:/var/www/ssc_update/

 


mkdir -p /var/www/ssc_update/

rm -rf /var/www/ssc_update/*

cp -rp /var/www/ssc/* /var/www/ssc_update/

tar zxvf /usr/local/shell/tar/ssc2012-07-11.tar.gz -C /var/www/ssc_update/

 

 

#进入主目录,不可缺少
cd /var/www

 

#制作补丁
diff -Nur ssc ssc_update > change.patch

 

#发布
patch -Np0 < change.patch

 

#还原
patch -Rp0 < change.patch 



 

====================  推荐 :理解 p0 和 p1

 

 

 

 

作为程序员,了解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

如果你想严格指定是应用补丁 可以使用下面命令(就是增加N参数):

# patch -N p0 < foo.patch


如果你想严格指定是还原补丁 可以使用下面命令(就是增加R参数):


# patch -R p0 < foo.patch


注释:在本例中,每次应用补丁后,自己还原补丁,以备后用继续试验,我就不多说了。

 

 


-----------看到这里如果你对patch的p参数还不太清楚的话,接着往下看,我们改变一下当前路径:

# cd old


此时就应该是p1,而不是p0了,引用foo.patch文件的路径也要相对变一下,因为当前目录已经是old了


# patch -p1 < ../foo.patch
patching file a/b/foo.txt

因为此时我们是在old下使用patch命令,和a子目录平级,而补丁文件foo.patch里的路径声明是:

--- old/a/b/foo.txt

也就是说第一个斜线左边的old/部分已经没用了,这就是p1的含义!

继续往深度变换路径,依次测试使用p2,p3参数:

# cd a

# patch -p2 < ../../foo.patch
patching file b/foo.txt

# cd b

# patch -p3 < ../../../foo.patch
patching file foo.txt

在本例中,p3已经是最深目录了,此时可以省略p参数:

# patch < ../../../foo.patch
patching file foo.txt

也就是说,不使用p参数的时候,patch命令会忽略任何目录,直接使用文件。

下面接着文章前面说的为什么使用diff命令时最好不要使用绝对路径,而应该使用相对路径?

答:如果你在使用diff的时候使用的是绝对路径,那么补丁文件里的文件路径信息会类似下面的样子:

--- /a/b/c/d/e/f/g/bar.txt

如此一来,当别人想应用你的补丁时,因为目录结构肯定有差异,所以就不得不费力判断到底使用p几。这样一来就很容易出错,相反,如果使用相对路径的话,大多数时候,p0或者p1就足够了,不易出错。

跟着本文的步骤操作一下,肯定能掌握diff&patch用法,基本上使用diff时就是"diff -Naur FROM TO"(FROM, TO为变量)这样的固定打法,然后在使用patch的时候,先看看补丁文件的大致内容,结合当前目录以确定需要跳过的目录数,然后套用"patch -pN < patch.file"(N为变量)即可。

 

-------------------

  总结一下:

  单个文件

  diff –uN from-file to-file >to-file.patch

  patch –p0 < to-file.patch

  patch –RE –p0 < to-file.patch

 

 

  多个文件

  diff –uNr from-docu to-docu >to-docu.patch

  patch –p1 < to-docu.patch

  patch –R –p1

  -------------------

 

应用

  为内核打补丁。前面在创建交叉编译工具链时,其中有一步就是为内核打补丁。当时还不是特别了解,现在很清晰了。参考前面的文章《基于ARM+Linux嵌入式开发的开发工具链的建立》。


  1、首先是解压,因为发布的补丁文件都是使用gzip压缩的。

  $gunzip ../setup-dir/ patch-2.4.21-rmk1.gz

  2、然后进入你的内核源代码目录

  $cd linux-2.4.21

  3、打补丁

  $patch –p1 < ../../setup-dir/patch-2.4.21-rmk1

  打完补丁后,需要检查一下有没有拒绝执行的文件,即检查.rej文件的存在。使用命令:

  $find . -name *.rej

  如果发现,会将其输出到标准输出终端,默认屏幕。当然,你也可以采用重定向,输出到指定文件,比如reject。

  $fine . -name *.rej >reject

  然后可以查看reject的内容了。

 

 

分享到:
评论

相关推荐

    google-diff-match-patch比较文件 demo

    要对文本文件的进行比较的时候,可以考虑使用google-diff-match-patch,它可以进行比较、匹配和生成补丁的操作 google-diff-match-patch这个类库提供了强大的算法用于纯文本内容的差异比较,匹配,打补丁,实现...

    diff命令 比较文件的差异

    如果指定要比较目录,则diff会比较目录中相同文件名的文件,但不会比较其中子目录 。 语法格式:diff [参数] [目录] 常用参数: -a diff预设只会逐行比较文本文件 -b 不检查空格字符的不同 -W 在使用-y...

    windows可用的diff和patch

    windows可用的diff和patch可执行文件,用于比较文件、生成补丁和应用补丁。用于一键给多个源文件加指定代码。比如应用systemview时需要对FreeRTOS进行的很多更改,参见...

    diff_match_patch库

    diff_match_patch java资源库,方便文件之间求差异,和根据差异推算出新的文件,主要用于增量更新

    如何使用diff和patch

    1) 在数学上来说:diff/patch,diff是对2个集合求差,patch是求和  diff A B &gt;C ,一般A是原始文件,B是修改后的文件,C称为A的补丁文件。  patch A C 就能得到B, 这一步叫做对A打上了B的名字为C的补丁。  patch ...

    diff_match_patch

    google-diff_match_patch:Java版本补丁包库文件

    diff和patch说明

    NULL 博文链接:https://chuqq.iteye.com/blog/481803

    java-diff源代码

    java-diff 源码

    前端HTML内容差异比较diff-match-patch.js

    前端HTML内容差异比较diff-match-patch.js

    prh_google_diff_match_patch.zip

    使用google-diff-match-patch比较文件 要对文本文件的进行比较的时候,可以考虑使用google-diff-match-patch,它可以进行比较、匹配和生成补丁的操作。

    jsdiff一个javascript的文本差异比较实现

    jsdiff 一个javascript的文本差异比较实现

    diff-match-patch-master.zip

    diff-match-patch

    diff工具(多文件夹比较,多文件比较,找不同)

    国外一款diff工具,简洁的英文界面。可比较两个文件夹的异同、两个文件的异同,适用于代码版本比较以及文件备份时的查缺补漏

    angular-diff-match-patch, 用于google差异匹配补丁的AngularJS包装器.zip

    angular-diff-match-patch, 用于google差异匹配补丁的AngularJS包装器 angular-diff-match-patch 这个库只是 google-diff-match-patch的一个包装器。 ( 这里显示一些自定义样式) Angular 2端口要在 Angular 2 项

    详解Linux patch命令参数及用法

    diff命令,在制作patch文件的时候,基本上只需要使用到diff -Nau 这个参数,如果比较的是文件夹,还要加上-r参数,所以一般直接使用Naur参数。 功能说明:修补文件。 语 法:patch [-bceEflnNRstTuvZ][-B &lt;备份...

    shell中常用的命令之diff和patch用法

    shell中常用的命令:diff(比较文件内容)和patch(打补丁) 1.diff用法 比较文件内容不同 diff file1 fil2 #以后面的文件file2作为标准 比较file1相对于file2来说有什么不同 比较文件例子 -r:比较目录 diff -r ...

    Shell编程中的常用命令diff和patch

    diff通常比较目录的时候 比较的结果是目录中文件的不同,不能比较文件内容的不同 (有没有相同名字的文件) a表示添加 —-add c表示更改 —-change d表示删除 —-delete   Diff file1 file2 #以第二个文件为基准,看...

    windiff文件/文件夹比较工具源代码

    这份代码虽然完整,但是无法用VS打开运行,功能可以参考压缩文件中的EXE程序,是网上下的

Global site tag (gtag.js) - Google Analytics