使用patch給代碼打補丁,快速高效、patch -p1 和p0 的區別

  • 生成patch:
git diff > file.patch
  • 打patch:
patch -p1 < file.patch
git apply file.patch
說到patch命令,就不得不提到diff命令,也就是製作patch的必要工具。diff命令,在製作patch文件的時候,基本上只需要使用到diff -Nau 這個參數,如果比較的是文件夾,還要加上-r參數,所以一般直接使用Naur參數。

實驗的基本步驟。我打算是建立一個級聯目錄./x/xx/xxx/,在xxx目錄下建立兩個不同的文件xxx1,xxx2。然後在xxx目錄下用diff命令,建立一個補丁文件xxx.patch,在xx目錄下建立一個補丁文件xx.patch,在x目錄下建立一個補丁文件x.patch。然後在這三個目錄下實驗。

開始實驗:建立實驗目錄
[King@Fedora ~]$ mkdir -pv x/xx/xxx
mkdir: 已創建目錄 “x”
mkdir: 已創建目錄 “x/xx”
mkdir: 已創建目錄 “x/xx/xxx”

進入xxx目錄下創建xxx1,xxx2

[King@Fedora ~]$ cd x/xx/xxx
[King@Fedora xxx]$ cat >> xxx1 << EOF
> 111111
> 111111
> EOF
[King@Fedora xxx]$ cat >> xxx2 << EOF
> 111111
> 222222
> EOF

查看這兩個文件:

[King@Fedora xxx]$ diff -y xxx1 xxx2
111111                                111111
111111                           |    222222

一定要注意:打補丁時所在的目錄

在xxx目錄下創建補丁文件xxx.patch,並查看。
[King@Fedora xxx]$ diff -Naru xxx1 xxx2 > xxx.patch
[King@Fedora xxx]$ cat xxx.patch 
- - - xxx1    2009-12-19 22:28:26.582959182 +0800
+++ xxx2    2009-12-19 22:28:42.798928591 +0800
@@ -1,2 +1,2 @@
  111111
- 111111
+222222

在xx目錄下創建補丁文件xx.patch,並查看:

[King@Fedora xxx]$ cd ..
[King@Fedora xx]$ diff -Naru xxx/xxx1 xxx/xxx2 > xx.patch
[King@Fedora xx]$ cat xx.patch 
--- xxx/xxx1    2009-12-19 22:28:26.582959182 +0800
+++ xxx/xxx2    2009-12-19 22:28:42.798928591 +0800
@@ -1,2 +1,2 @@
111111
-111111
+222222

在x目錄下創建補丁文件x.patch,並查看

[King@Fedora xx]$ cd ..
[King@Fedora x]$ diff -Nu xx/xxx/xxx1 xx/xxx/xxx2 > x.patch
[King@Fedora x]$ cat x.patch 
--- xx/xxx/xxx1    2009-12-19 22:28:26.582959182 +0800
+++ xx/xxx/xxx2    2009-12-19 22:28:42.798928591 +0800
@@ -1,2 +1,2 @@
111111
-111111
+222222

現將patch文件都拷貝到xxx目錄下去。

[King@Fedora x]$ cp x.patch xx/xxx/
[King@Fedora x]$ cp xx/xx.patch xx/xxx/

進入xxx目錄開始實驗

[King@Fedora x]$ cd xx/xxx
[King@Fedora xxx]$ ls
x.patch  xx.patch  xxx1  xxx2  xxx.patch
[King@Fedora xxx]$ patch-p0< xxx.patch  #用第二個的 補丁 修改 第一個文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE < xxx.patch #用第一個的 補丁 修改 第一個文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111
[King@Fedora xxx]$ patch -p1 < xx.patch
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE < xxx.patch
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111
[King@Fedora xxx]$ patch -p2 < x.patch
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE < x.patch
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111
[King@Fedora xx]$ patch-p0 < xx.patch  # 用第二個的 補丁 修改 第一個文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE < xxx.patch #用第一個的 補丁 修改 第一個文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111

[King@Fedora xxx]$ patch -p1 < x.patch
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE < xxx.patch
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111
[King@Fedora x]$ patch-p0< x.patch  # 用第二個的 補丁 修改 第一個文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE < xxx.patch #用第一個的 補丁 修改 第一個文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111

這裏唯一需要說明的是p0的含義,因爲在x.patch補丁文件裏的路徑信息是這樣的:

--- xx/xxx/xxx1   

**p表示跳過幾級目錄.**因爲是在x目錄下使用的patch命令,xx目錄就在x目錄下,所以不必跳過任何目錄,而應該使用— xx/xxx/xxx1 完整路徑,所以此時使用的是p0。

注意:patch -p後面是不能帶負數 的。不使用p參數的時候,patch命令會 忽略 任何目錄,直接使用文件。

[King@Fedora x]$ patch x/xx/xxx/xxx1 < x.patch # 用補丁x.patch 直接修改 文件xxx1,因爲沒有用p參數,所以 會 忽略掉補丁文件裏的 所有目錄。

作爲程序員,瞭解diff&patch命令是非常必要的。比如說我們發現某個項目有bug代碼,而自己又沒有提交權限,那麼此時最合適的解決方法就是用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

內容來自Linuxren.net
其中-Naur參數屬於固定用法,大多數時候,在使用diff命令時搭配這個參數就可以了。

大概瀏覽一下補丁文件:

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命令,系統會問你是否想還原,輸入y 還原:

# 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 -Np0 < foo.patch

如果你想嚴格指定是 還原補丁 可以使用下面命令(就是增加R參數):

# patch -Rp0 < foo.patch

註釋:在本例中,每次應用補丁後,自己還原補丁,以備後用繼續試驗,我就不多說了。

看到這裏如果你對patch的p參數還不太清楚的話,接着往下看,我們改變一下當前路徑:

# cd old

此時就應該是p1,而不是p0了,引用foo.patch文件的路徑也要相對變一下,因爲當前目錄已經是old了: Linuxren.net

# 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就足夠了,不易出錯。

百度百科得出的一些用途解釋:
Linux diff命令用於比較文件的差異:
diff以逐行的方式,比較文本文件的異同處。如果指定要比較目錄,則diff會比較目錄中相同文件名的文件,但不會比較其中子目錄。

語法
diff [-abBcdefHilnNpPqrstTuvwy][-<行數>][-C <行數>][-D <巨集名稱>][-I <字符或字符串>][-S <文件>][-W <寬度>][-x <文件或目錄>][-X <文件>][–help][–left-column][–suppress-common-line][文件或目錄1][文件或目錄2]
參數:

-<行數>  指定要顯示多少行的文本。此參數必須與-c或-u參數一併使用。-a或–text  diff預設只會逐行比較文本文件。-b或–ignore-space-change  不檢查空格字符的不同。

-B或–ignore-blank-lines  不檢查空白行。
-c  顯示全部內文,並標出不同之處。
-C<行數>或–context<行數>  與執行"-c-<行數>"指令相同。
-d或–minimal  使用不同的演算法,以較小的單位來做比較。
-D<巨集名稱>或ifdef<巨集名稱>  此參數的輸出格式可用於前置處理器巨集。
-e或–ed  此參數的輸出格式可用於ed的script文件。
-f或-forward-ed  輸出的格式類似ed的script文件,但按照原來文件的順序來顯示不同處。
-H或–speed-large-files  比較大文件時,可加快速度。
-l<字符或字符串>或–ignore-matching-lines<字符或字符串>  若兩個文件在某幾行有所不同,而這幾行同時都包含了選項中指定的字符或字符串,則不顯示這兩個文件的差異。
-i或–ignore-case  不檢查大小寫的不同。
-l或–paginate  將結果交由pr程序來分頁。
-n或–rcs  將比較結果以RCS的格式來顯示。
-N或–new-file  在比較目錄時,若文件A僅出現在某個目錄中,預設會顯示:
Only in目錄:文件A若使用-N參數,則diff會將文件A與一個空白的文件比較。
-p  若比較的文件爲C語言的程序碼文件時,顯示差異所在的函數名稱。
-P或–unidirectional-new-file  與-N類似,但只有當第二個目錄包含了一個第一個目錄所沒有的文件時,纔會將這個文件與空白的文件做比較。
-q或–brief  僅顯示有無差異,不顯示詳細的信息。
-r或–recursive  比較子目錄中的文件。
-s或–report-identical-files  若沒有發現任何差異,仍然顯示信息。
-S<文件>或–starting-file<文件>  在比較目錄時,從指定的文件開始比較。
-t或–expand-tabs  在輸出時,將tab字符展開。
-T或–initial-tab  在每行前面加上tab字符以便對齊。
-u,-U<列數>或–unified=<列數>  以合併的方式來顯示文件內容的不同。
-v或–version  顯示版本信息。
-w或–ignore-all-space  忽略全部的空格字符。
-W<寬度>或–width<寬度>  在使用-y參數時,指定欄寬。
-x<文件名或目錄>或–exclude<文件名或目錄>  不比較選項中所指定的文件或目錄。
-X<文件>或–exclude-from<文件>  您可以將文件或目錄類型存成文本文件,然後在=<文件>中指定此文本文件。
-y或–side-by-side  以並列的方式顯示文件的異同之處。
–help  顯示幫助。
–left-column  在使用-y參數時,若兩個文件某一行內容相同,則僅在左側的欄位顯示該行內容。
–suppress-common-lines  在使用-y參數時,僅顯示不同之處。

Linux patch命令用於修補文件。
patch指令讓用戶利用設置修補文件的方式,修改,更新原始文件。倘若一次僅修改一個文件,可直接在指令列中下達指令依序執行。如果配合修補文件的方式則能一次修補大批文件,這也是Linux系統核心的升級方法之一。

語法
patch [-bceEflnNRstTuvZ][-B <備份字首字符串>][-d <工作目錄>][-D <標示符號>][-F <監別列數>][-g <控制數值>][-i <修補文件>][-o <輸出文件>][-p <剝離層級>][-r <拒絕文件>][-V <備份方式>][-Y <備份字首字符串>][-z <備份字尾字符串>][–backup-if -mismatch][–binary][–help][–nobackup-if-mismatch][–verbose][原始文件 <修補文件>] 或 path [-p <剝離層級>] < [修補文件]
參數:

-b或–backup  備份每一個原始文件。
-B<備份字首字符串>或–prefix=<備份字首字符串>  設置文件備份時,附加在文件名稱前面的字首字符串,該字符串可以是路徑名稱。
-c或–context  把修補數據解譯成關聯性的差異。
-d<工作目錄>或–directory=<工作目錄>  設置工作目錄。
-D<標示符號>或–ifdef=<標示符號>  用指定的符號把改變的地方標示出來。
-e或–ed  把修補數據解譯成ed指令可用的敘述文件。
-E或–remove-empty-files  若修補過後輸出的文件其內容是一片空白,則移除該文件。
-f或–force  此參數的效果和指定"-t"參數類似,但會假設修補數據的版本爲新 版本。
-F<監別列數>或–fuzz<監別列數>  設置監別列數的最大值。
-g<控制數值>或–get=<控制數值>  設置以RSC或SCCS控制修補作業。
-i<修補文件>或–input=<修補文件>  讀取指定的修補問家你。
-l或–ignore-whitespace  忽略修補數據與輸入數據的跳格,空格字符。
-n或–normal  把修補數據解譯成一般性的差異。
-N或–forward  忽略修補的數據較原始文件的版本更舊,或該版本的修補數據已使 用過。
-o<輸出文件>或–output=<輸出文件>  設置輸出文件的名稱,修補過的文件會以該名稱存放。
-p<剝離層級>或–strip=<剝離層級>  設置欲剝離幾層路徑名稱。
-f<拒絕文件>或–reject-file=<拒絕文件>  設置保存拒絕修補相關信息的文件名稱,預設的文件名稱爲.rej。
-R或–reverse  假設修補數據是由新舊文件交換位置而產生。
-s或–quiet或–silent  不顯示指令執行過程,除非發生錯誤。
-t或–batch  自動略過錯誤,不詢問任何問題。
-T或–set-time  此參數的效果和指定"-Z"參數類似,但以本地時間爲主。
-u或–unified  把修補數據解譯成一致化的差異。
-v或–version  顯示版本信息。
-V<備份方式>或–version-control=<備份方式>  用"-b"參數備份目標文件後,備份文件的字尾會被加上一個備份字符串,這個字符串不僅可用"-z"參數變更,當使用"-V"參數指定不同備份方式時,也會產生不同字尾的備份字符串。
-Y<備份字首字符串>或–basename-prefix=–<備份字首字符串>  設置文件備份時,附加在文件基本名稱開頭的字首字符串。
-z<備份字尾字符串>或–suffix=<備份字尾字符串>  此參數的效果和指定"-B"參數類似,差別在於修補作業使用的路徑與文件名若爲src/linux/fs/super.c,加上"backup/"字符串後,文件super.c會備份於/src/linux/fs/backup目錄裏。
-Z或–set-utc  把修補過的文件更改,存取時間設爲UTC。
–backup-if-mismatch  在修補數據不完全吻合,且沒有刻意指定要備份文件時,才備份文件。
–binary  以二進制模式讀寫數據,而不通過標準輸出設備。
–help  在線幫助。
–nobackup-if-mismatch  在修補數據不完全吻合,且沒有刻意指定要備份文件時,不要備份文件。
–verbose  詳細顯示指令的執行過程

發佈了213 篇原創文章 · 獲贊 239 · 訪問量 88萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章