使用diff和patch進行簡單的文件版本管理

在Unix系統下,維護源碼版本可以使用很多方法,其中最常用的當然是大名鼎鼎的CVS,但實際上,簡單的版本維護工作並沒有必要使用複雜的CVS等專門的版本維護工具,Unix標配中的diff和patch工具就完全可以完成代碼的簡單備份和升級工作。

diff 以"行"爲單位比較兩個文本文件(也可以是目錄比較),並將不同之處以某種格式輸出到標準輸出上;patch可以讀入這種輸出,並按照一定指令使源文件 (目錄)按照目標文件(目錄)更新。Linux內核源碼就是按照這種方式保持更新的,我們在www.kernel.org上可以下載到最新內核的 patch文件的bzip2包。本文以gnudiffutils 2.7和patch 2.5爲例介紹diff和patch工具的使用。

1.diff

diff 既可以用來比較兩個文件,也可以用來比較兩個目錄中每個文件。使用-r(--recursive)參數時還可以在目錄中嵌套比較。比較目錄時除比較同名文 件外,對不同名的文件當成新文件處理。對於比較C程序文件,diff還提供了專門的參數(-p,--show-c-function)來標識不同之處所在 的函數名。

diff的輸出格式有三種:列舉方式、命令模式和上下文模式,其中命令模式有分爲兩種:ed命令格式和RCS(Revision Control System,版本控制系統)命令格式,上下文模式也按格式分爲老版和新版兩種。看下面的例子就能基本清楚各個格式的區別:



命令格式記錄的是從test1更新到test2所需要執行的命令,而上下文模式通常可讀性更好一些,它所記錄的主要是二者的差異,通常還記錄所需修改部分的上下幾行(可配置)內容以供比較。見下面的例子:



新版格式較之老版要緊湊一些,Linux內核源碼的升級就是按照新版上下文格式用diff組織的,比如patch-2.4.16中所用的具體命令爲:

diff -Nur linux-2.4.15 linux

參數N表示如果某個文件僅在一個目錄中出現,則假定其在另一個目錄中爲空文件;u表示unified格式,r表示在目錄中嵌套使用,linux-2.4.15顯然是老核的目錄名,而linux則爲新核的目錄名。

 2 patch

盡 管並沒有指定patch和diff的關係,但通常patch都使用diff的結果來完成打補丁的工作,這和patch本身支持多種diff輸出文件格式有 很大關係。patch通過讀入patch命令文件(可以從標準輸入),對目標文件進行修改。通常先用diff命令比較新老版本,patch命令文件則採用 diff的輸出文件,從而保持原版本與新版本一致。

patch的標準格式爲


patch [options] [originalfile] [patchfile]

如果patchfile爲空則從 標準輸入讀取patchfile內容;如果originalfile也爲空,則從patchfile(肯定來自標準輸入)中讀取需要打補丁的文件名。因 此,如果需要修改的是目錄,一般都必須在patchfile中記錄目錄下的各個文件名。絕大多數情況下,patch都用以下這種簡單的方式使用:


patch -p[num] <patchfile

patch命令可以忽略文件中的 冗餘信息,從中取出diff的格式以及所需要patch的文件名,文件名按照diff參數中的"源文件"、"目標文件"以及冗餘信息中的"Index:" 行中所指定的文件的順序來決定。也就是說,對於如下diff結果文件(Linux內核源碼2.4.16升級包,部分):

 
diff -Nur linux-2.4.15/Makefile linux/Makefile
--- linux-2.4.15/Makefile Thu Nov 22 17:22:58 2001
+++ linux/Makefile Sat Nov 24 16:21:53 2001
@@ -1,7 +1,7 @@
VERSION = 2
PATCHLEVEL = 4
-SUBLEVEL = 15
-EXTRAVERSION =-greased-turkey
+SUBLEVEL = 16
+EXTRAVERSION =

KERNELRELEASE=$(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)
……

patch首先嚐試當前目錄(或 者-d參數指定的目錄)下的linux-2.4.15/Makefile文件是否存在,如果不存在則試圖對linux/Makefile文件操作,僅當兩 者都不存在時(或者設置了POSIXLY_CORRECT環境變量)纔會讀取Index:的內容(此文件中沒有標識)。

前 面提到的-p參數決定了是否使用讀出的源文件名的前綴目錄信息,不提供-p參數,則忽略所有目錄信息,-p0(或者-p 0)表示使用全部的路徑信息,-p1將忽略第一個"/"以前的目錄,依此類推。如/usr/src/linux-2.4.15/Makefile這樣的文 件名,在提供-p3參數時將使用linux-2.4.15/Makefile作爲所要patch的文件。

對於剛纔舉的 Linux內核源碼2.4.16升級包的例子,假定源碼目錄位於/usr/src/linux中,則在當前目錄爲/usr/src時使用"patch -p0 <patch-2.4.16"可以工作,在當前目錄爲/usr/src/linux時,"patch -p1<patch-2.4.16"也可以正常工作。

patch可以直接操作上下文格式以及混合ed格式的diff輸出文件,而將ed格式文件通過管道提交給ed程序操作(暫時不知RCS格式的文件如何處理)。


3. 配合使用diff和patch升級源碼

在此僅舉一個簡單的例子來說明如何用diff/patch工具維護源碼升級。

假設program-1.0目錄中爲老版,現開發完成的新版位於program-2.0目錄中,將兩個目錄置於同一父目錄下,然後在該父目錄上執行:


diff -Nur program-1.0 program-2.0 >program-2.0.patch

將生成一個program-2.0.patch的補丁文件,發佈該補丁文件(當然可以先壓縮成bzip2格式)。

假設拿到的是program-2.0.patch.bz2文件,則在program-1.0目錄同級執行:


bzcat program-2.0.patch.bz2 | patch -p0

如此即完成了從1.0到2.0的升級。

如果希望恢復到原版本,可以使用-R(--reverse)參數,但僅對上下文格式的diff文件有效。還有一個備份參數也可以使用,但簡單應用中,整個目錄備份可能更方便一些。


 


 


一個例子:

fft爲當前工作目錄,fft.bak爲源文件備份。
現在diff -Nur fft fft.bak > diff.fft,則可以生成一個關於fft的diff文件。這個文件的目標修改目錄是fft.bak
如果執行patch -p0 < diff.fft,則可以將fft.bak中的文件升級爲fft
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章