作為程序員,了解diff&patch命 令是非常必要的。比如說我們發(fā)現(xiàn)某個項目有bug代碼,而自己又沒有svn的提交權限,那么此時最合適的解決方法就是用diff命令做一個補丁發(fā)給項目成 員。項目成員通過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 假設我們發(fā)現(xiàn)項目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環(huán)境變量: # diff -Naur old new > foo.patch 其中-Naur參數(shù)屬于固定打法,不管是對一個文件,還是對一個目錄,在使用這個參數(shù)基本就可以了。 大概瀏覽一下補丁文件: # 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 加減號后面的內(nèi)容是有用的內(nèi)容,其他的內(nèi)容是方便你查閱的相關信息內(nèi)容,補丁制作完成。 此時的文件目錄結構大概如下所示: #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。 查看一下目標文件,你會發(fā)現(xiàn)內(nèi)容已經(jīng)修改成新的了: # cat old/a/b/foo.txt new_line_1 new_line_2 此時如果你再次使用patch命令,系統(tǒng)會問你是否想還原: # patch -p0 < foo.patch patching file old/a/b/foo.txt Reversed (or previously applied) patch detected! Assume -R? [n] y 查看一下目標文件,你會發(fā)現(xiàn)內(nèi)容已經(jīng)還原成舊的了: # cat old/a/b/foo.txt old_line_1 old_line_2 如果你想嚴格指定是應用補丁可以使用下面命令(就是增加N參數(shù)): # patch -Np0 < foo.patch 如果你想嚴格指定是還原補丁可以使用下面命令(就是增加R參數(shù)): # patch -Rp0 < foo.patch 注釋:在本例中,每次應用補丁后,自己還原補丁,以備后用繼續(xù)試驗,我就不多說了。 看到這里如果你對patch的p參數(shù)還不太清楚的話,接著往下看,我們改變一下當前路徑: # cd old 此時就應該是p1,而不是p0了,引用foo.patch文件的路徑也要相對變一下,因為當前目錄已經(jīng)是old了: # patch -p1 < ../foo.patch patching file a/b/foo.txt 因為此時我們是在old下使用patch命令,和a子目錄平級,而補丁文件foo.patch里的路徑聲明是: --- old/a/b/foo.txt 也就是說第一個斜線左邊的old/部分已經(jīng)沒用了,這就是p1的含義! 繼續(xù)往深度變換路徑,依次測試使用p2,p3參數(shù): # cd a # patch -p2 < ../../foo.patch patching file b/foo.txt # cd b # patch -p3 < ../../../foo.patch patching file foo.txt 在本例中,p3已經(jīng)是最深目錄了,此時可以省略p參數(shù): # patch < ../../../foo.patch patching file foo.txt 也就是說,不使用p參數(shù)的時候,patch命令會忽略任何目錄,直接使用文件。 下面接著文章前面說的為什么使用diff命令時最好不要使用絕對路徑,而應該使用相對路徑? 答:如果你在使用diff的時候使用的是絕對路徑,那么補丁文件里的文件路徑信息會類似下面的樣子: --- /a/b/c/d/e/f/g/bar.txt 如此一來,當別人想應用你的補丁時,因為目錄結構肯定有差異,所以就不得不費力判斷到底使用p幾。這樣一來就很容易出錯,相反,如果使用相對路徑的話,大多數(shù)時候,p0或者p1就足夠了,不易出錯。 跟著本文的步驟操作一下,肯定能掌握diff&patch用法,基本上使用diff時就是"diff -Naur FROM TO"(FROM, TO為變量)這樣的固定打法,然后在使用patch的時候,先看看補丁文件的大致內(nèi)容,結合當前目錄以確定需要跳過的目錄數(shù),然后套用"patch -pN < patch.file"(N為變量)即可。 ------------------- diff –uNr from-docu to-docu >to-docu.patch patch –p1 < to-docu.patch patch –R –p1 ?。?/p> 應用 |
|
來自: raymoon_sure > 《linux》