一、原理
diff比較兩個(gè)文件或文件集合的差異,并記錄下來,生成一個(gè)diff文件,這也是我們常說的patch文件,即補(bǔ)丁文件。
patch能將diff文件運(yùn)用于 原來的兩個(gè)集合之一,從而得到另一個(gè)集合。
舉個(gè)例子來說文件A和文件B,經(jīng)過diff之后生成了補(bǔ)丁文件C,那么著個(gè)過程相當(dāng)于 A -B = C ,那么patch的過程就是B+C = A 或A-C =B。
因此我們只要能得到A, B, C三個(gè)文件中的任何兩個(gè),就能用diff和patch這對(duì)工具生成另外一個(gè)文件。
二、用法
1. diff的用法
diff后面可以接兩個(gè)文件名或兩個(gè)目錄名。 如果是一個(gè)目錄名加一個(gè)文件名,那么只作用在那么個(gè)目錄下的同名文件。
如果是兩個(gè)目錄的話,作用于該目錄下的所有文件,不遞歸。如果我們希望遞歸執(zhí)行,需要使用-r參數(shù)。
命令diff A B >C ,一般A是原始文件,B是修改后的文件,C稱為A的補(bǔ)丁文件。
不加任何參數(shù)生成的diff文件格式是一種簡單的格式,這種格式只標(biāo)出了不一樣的行數(shù)和內(nèi)容。我們需要一種更詳細(xì)的格式,可以標(biāo)識(shí)出不同之處的上下文環(huán)境,這樣更有利于提高patch命令的識(shí)別能力。這個(gè)時(shí)候可以用-c開關(guān)。通常的參數(shù)為 -Nuar。
2. patch的用法
patch用于根據(jù)原文件和補(bǔ)丁文件生成目標(biāo)文件。還是拿上個(gè)例子來說
patch A C 就能得到B, 這一步叫做對(duì)A打上了B的名字為C的補(bǔ)丁。
之一步之后,你的文件A就變成了文件B。如果你打完補(bǔ)丁之后想恢復(fù)到A怎么辦呢?
patch -R B C 就可以重新還原到A了。
所以不用擔(dān)心會(huì)失去A的問題。
其實(shí)patch在具體使用的時(shí)候是不用指定原文件的,因?yàn)檠a(bǔ)丁文件中都已經(jīng)記載了原文件的路徑和名稱。 patch足夠聰明可以認(rèn)出來。但是有時(shí)候會(huì)有點(diǎn)小問題。比如一般對(duì)兩個(gè)目錄diff的時(shí)候可能已經(jīng)包含了原目錄的名字,但是我們打補(bǔ)丁的時(shí)候會(huì)進(jìn)入到目 錄中再使用patch,著個(gè)時(shí)候就需要你告訴 patch命令怎么處理補(bǔ)丁文件中的路徑??梢岳?pn開關(guān),告訴patch命令忽略的路徑分隔符的個(gè)數(shù)。舉例如下:
A文件在 DIR_A下,修改后的B文件在DIR_B下,一般DIR_A和DIR_B在同一級(jí)目錄。我們?yōu)榱藢?duì)整個(gè)目錄下的所有文件一次性diff,我們一般會(huì)到DIR_A和DIR_B的父目錄下執(zhí)行以下命令
diff -rc DIR_A DIR_B >C
這個(gè)時(shí)候補(bǔ)丁文件C中會(huì)記錄了原始文件的路徑為 DIR_A/A
現(xiàn)在另一個(gè)用戶得到了A文件和C文件,其中A文件所在的目錄也是DIR_A。 一般,他會(huì)比較喜歡在DIR_A目錄下面進(jìn)行patch操作,它會(huì)執(zhí)行
patch
但是這個(gè)時(shí)候patch分析C文件中的記錄,認(rèn)為原始文件是 DIR_A/A,但實(shí)際上是./A,此時(shí)patch會(huì)找不到原始文件。為了避免這種情況我們可以使用-p1參數(shù)如下
patch -p1
此時(shí),patch會(huì)忽略掉第1個(gè)”/”之前的內(nèi)容,認(rèn)為原始文件是 ./A,這樣就正確了。
最后有以下幾點(diǎn)注意:
1. 一次打多個(gè)patch的話,一般這些patch有先后順序,得按次序打才行。
2. 在patch之前不要對(duì)原文件進(jìn)行任何修改
3. 如果patch中記錄的原始文件和你得到的原始文件版本不匹配(很容易出現(xiàn)),那么你可以嘗試使用patch, 如果幸運(yùn)的話,可以成功。大部分情況下,會(huì)有不匹配的情況,此時(shí)patch會(huì)生成rej文件,記錄失敗的地方,你可以手工修改。
三、舉例
通常情況下,diff與patch一起使用。即使用patch調(diào)用diff文件。如下面這個(gè)例子:
有兩個(gè)目錄為old和new,old中有f1.c,f2.c;new中有f1.c,f3.c?,F(xiàn)要將new補(bǔ)到old上。
[fsy@localhost ~]$ mkdir old new
[fsy@localhost ~]$ cd old
[fsy@localhost old]$ touch f1.c f2.c
###建立兩個(gè)文件夾和old中的文件
[fsy@localhost old]$ echo 1 > f1.c
[fsy@localhost old]$ cat f1.c
1
[fsy@localhost old]$ echo 2 > f2.c
###在old的文件中寫入內(nèi)容
[fsy@localhost old]$ cd ../new
[fsy@localhost new]$ touch f1.c f3.c
[fsy@localhost new]$ echo new 1 > f1.c
[fsy@localhost new]$ cat f1.c
new 1
[fsy@localhost new]$ echo 3 > f3.c
###建立new文件并寫入內(nèi)容
[fsy@localhost new]$ cd ..
[fsy@localhost ~]$ diff -Nuar old new >dir.diff
[fsy@localhost ~]$ cat dir.diff
diff -Nuar ./old/f1.c ./new/f1.c
--- old/f1.c 2011-08-30 12:32:21.553737454 +0800
+++ new/f1.c 2011-08-30 12:34:28.444148124 +0800
@@ -1 +1 @@
-1
+new 1
diff -Nuar ./old/f2.c ./new/f2.c
--- old/f2.c 2011-08-30 12:32:40.334207279 +0800
+++ new/f2.c 1970-01-01 08:00:00.000000000 +0800
@@ -1 +0,0 @@
-2
diff -Nuar ./old/f3.c ./new/f3.c
--- old/f3.c 1970-01-01 08:00:00.000000000 +0800
+++ new/f3.c 2011-08-30 12:34:42.331754293 +0800
@@ -0,0 +1 @@
+3
###可以清楚的看見兩個(gè)文件夾下文件的不同之處
[fsy@localhost ~]$ cd old ..........................................必須進(jìn)入到要打補(bǔ)丁的文件夾
[fsy@localhost old]$ patch -p1 ../dir.diff
patching file old/f1.c
patching file old/f2.c
patching file old/f3.c
[fsy@localhost old]$ ls
f1.c f3.c
[fsy@localhost old]$ cat f1.c
new 1
[fsy@localhost old]$ cat f3.c
3
[fsy@localhost old]$ /p>
p>###f2.c被刪除,f1.c和f3.c被更新。/p>
p>