說到patch命令,就不得不提到diff命令,也就是制作patch的必要工具。diff命令,在制作patch文件的時(shí)候,基本上只需要使用到diff -Nau 這個(gè)參數(shù),如果比較的是文件夾,還要加上-r參數(shù),所以一般直接使用Naur參數(shù)。
實(shí)驗(yàn)的基本步驟。我打算是建立一個(gè)級聯(lián)目錄./x/xx/xxx/,在xxx目錄下建立兩個(gè)不同的文件xxx1,xxx2。然后在xxx目錄下用diff命令,建立一個(gè)補(bǔ)丁文件xxx.patch,在xx目錄下建立一個(gè)補(bǔ)丁文件xx.patch,在x目錄下建立一個(gè)補(bǔ)丁文件x.patch。然后在這三個(gè)目錄下實(shí)驗(yàn)。
開始實(shí)驗(yàn):建立實(shí)驗(yàn)?zāi)夸?br />
[King@Fedora ~]$ mkdir -pv x/xx/xxx
mkdir: 已創(chuàng)建目錄 “x”
mkdir: 已創(chuàng)建目錄 “x/xx”
mkdir: 已創(chuàng)建目錄 “x/xx/xxx”
進(jìn)入xxx目錄下創(chuàng)建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
查看這兩個(gè)文件
[King@Fedora xxx]$ diff -y xxx1 xxx2
111111 111111
111111 | 222222
一定要注意:打補(bǔ)丁時(shí)所在的目錄
在xxx目錄下創(chuàng)建補(bǔ)丁文件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目錄下創(chuàng)建補(bǔ)丁文件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目錄下創(chuàng)建補(bǔ)丁文件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
現(xiàn)將patch文件都拷貝到xxx目錄下去。
[King@Fedora x]$ cp x.patch xx/xxx/
[King@Fedora x]$ cp xx/xx.patch xx/xxx/
進(jìn)入xxx目錄開始實(shí)驗(yàn)
[King@Fedora x]$ cd xx/xxx
[King@Fedora xxx]$ ls
x.patch xx.patch xxx1 xxx2 xxx.patch
[King@Fedora xxx]$ patch-p0xxx.patch #用第二個(gè)的 補(bǔ)丁 修改 第一個(gè)文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE xxx.patch #用第一個(gè)的 補(bǔ)丁 修改 第一個(gè)文件
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 # 用第二個(gè)的 補(bǔ)丁 修改 第一個(gè)文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE xxx.patch #用第一個(gè)的 補(bǔ)丁 修改 第一個(gè)文件
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-p0x.patch # 用第二個(gè)的 補(bǔ)丁 修改 第一個(gè)文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
222222
[King@Fedora xxx]$ patch -RE xxx.patch #用第一個(gè)的 補(bǔ)丁 修改 第一個(gè)文件
patching file xxx1
[King@Fedora xxx]$ cat xxx1
111111
111111
這里唯一需要說明的是p0的含義,因?yàn)樵趚.patch補(bǔ)丁文件里的路徑信息是這樣的:
--- xx/xxx/xxx1
p表示跳過幾級目錄,因?yàn)槭窃趚目錄下使用的patch命令,xx目錄就在x目錄下,所以不必跳過任何目錄,而應(yīng)該使用--- xx/xxx/xxx1 完整路徑,所以此時(shí)使用的是p0。
注意:patch -p后面是不能帶負(fù)數(shù) 的。不使用p參數(shù)的時(shí)候,patch命令會(huì) 忽略 任何目錄,直接使用文件。
[King@Fedorax]$ patch x/xx/xxx/xxx1 x.patch # 用補(bǔ)丁x.patch 直接修改 文件xxx1,因?yàn)闆]有用p參數(shù),所以 會(huì) 忽略掉補(bǔ)丁文件里的 所有目錄。
作為程序員,了解diffpatch命令是非常必要的。比如說我們發(fā)現(xiàn)某個(gè)項(xiàng)目有bug代碼,而自己又沒有提交權(quán)限,那么此時(shí)最合適的解決方法就是用diff命令做一個(gè)補(bǔ)丁發(fā)給項(xiàng)目成員。項(xiàng)目成員通過patch命令可以立刻知道你的意圖。有人會(huì)說直接傳一個(gè)新文件不是更簡單?不要忘了,一個(gè)patch文件尺寸更小傳輸更快,而且可以明顯的看到都做了哪些修改。
保證當(dāng)前目錄是demo名錄:
# mkdir demo
# cd demo
先模擬一個(gè)項(xiàng)目目錄old:
# mkdir -p old/a/b
# vi old/a/b/foo.txt
old_line_1
old_line_2
假設(shè)我們發(fā)現(xiàn)項(xiàng)目old有bug代碼,下面我們先拷貝一個(gè)新目錄new,并在此修改bug代碼:
# cp -r old new
# vi new/a/b/foo.txt
new_line_1
new_line_2
保證old和new兩個(gè)目錄都在當(dāng)前目錄下,下面就可以使用diff命令了,不要使用絕對路徑,而應(yīng)該使用相對路徑,至于原因,看到文章結(jié)尾你就清楚了:
# LC_ALL=C TZ=UTC0 diff -Naur old new > foo.patch
如果不在意字符集,時(shí)差等問題,也可以省略LC_ALL=C TZ=UTC0環(huán)境變量:
# diff -Naur old new > foo.patch
內(nèi)容來自Linuxren.net
其中-Naur參數(shù)屬于固定用法,大多數(shù)時(shí)候,在使用diff命令時(shí)搭配這個(gè)參數(shù)就可以了。
大概瀏覽一下補(bǔ)丁文件:
# 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)容是方便你查閱的相關(guān)信息內(nèi)容,補(bǔ)丁制作完成。
此時(shí)的文件目錄結(jié)構(gòu)大概如下所示:
#tree
demo
|-- old
| `-- a
| `-- b
| `-- foo.txt
|-- new
| `-- a
| `-- b
| `-- foo.txt
-- foo.patch
下面看看如何使用patch來應(yīng)用補(bǔ)丁,要注意的是當(dāng)前目錄是demo,試試下面命令:
# patch -p0 foo.patch
patching file old/a/b/foo.txt
這里唯一需要說明的是p0的含義,因?yàn)樵趂oo.patch補(bǔ)丁文件里的路徑信息是這樣的:
--- old/a/b/foo.txt
p表示跳過幾級目錄,因?yàn)槭窃赿emo目錄下使用的patch命令,old目錄就在demo目錄下,所以不必跳過任何目錄,而應(yīng)該使用old/a/b/foo.txt完整路徑,所以此時(shí)使用的是p0。
查看一下目標(biāo)文件,你會(huì)發(fā)現(xiàn)內(nèi)容已經(jīng)修改成新的了:
# cat old/a/b/foo.txt
new_line_1
new_line_2
此時(shí)如果你再次使用patch命令,系統(tǒng)會(huì)問你是否想還原,輸入y 還原:
# patch -p0 foo.patch
patching file old/a/b/foo.txt
Reversed (or previously applied) patch detected! Assume -R? [n] y
查看一下目標(biāo)文件,你會(huì)發(fā)現(xiàn)內(nèi)容已經(jīng)還原成舊的了:
# cat old/a/b/foo.txt
old_line_1
old_line_2
如果你想嚴(yán)格指定是 應(yīng)用補(bǔ)丁 可以使用下面命令(就是增加N參數(shù)):
# patch -Np0 foo.patch
如果你想嚴(yán)格指定是 還原補(bǔ)丁 可以使用下面命令(就是增加R參數(shù)):
# patch -Rp0 foo.patch
注釋:在本例中,每次應(yīng)用補(bǔ)丁后,自己還原補(bǔ)丁,以備后用繼續(xù)試驗(yàn),我就不多說了。
看到這里如果你對patch的p參數(shù)還不太清楚的話,接著往下看,我們改變一下當(dāng)前路徑:
# cd old
此時(shí)就應(yīng)該是p1,而不是p0了,引用foo.patch文件的路徑也要相對變一下,因?yàn)楫?dāng)前目錄已經(jīng)是old了: Linuxren.net
# patch -p1 ../foo.patch
patching file a/b/foo.txt
因?yàn)?strong>此時(shí)我們是在old下使用patch命令,和a子目錄平級,而補(bǔ)丁文件foo.patch里的路徑聲明是:
--- old/a/b/foo.txt
也就是說第一個(gè)斜線左邊的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)是最深目錄了,此時(shí)可以省略p參數(shù):
# patch ../../../foo.patch
patching file foo.txt
也就是說,不使用p參數(shù)的時(shí)候,patch命令會(huì) 忽略 任何目錄,直接使用文件。
下面接著文章前面說的為什么使用diff命令時(shí)最好不要使用絕對路徑,而應(yīng)該使用相對路徑?
答:如果你在使用diff的時(shí)候使用的是絕對路徑,那么補(bǔ)丁文件里的文件路徑信息會(huì)類似下面的樣子:
--- /a/b/c/d/e/f/g/bar.txt
如此一來,當(dāng)別人想應(yīng)用你的補(bǔ)丁時(shí),因?yàn)槟夸浗Y(jié)構(gòu)肯定有差異,所以就不得不費(fèi)力判斷到底使用p幾。這樣一來就很容易出錯(cuò),相反,如果使用相對路徑的話,大多數(shù)時(shí)候,p0或者p1就足夠了,不易出錯(cuò)。
以上所述是小編給大家介紹的linux patch 命令小結(jié),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
您可能感興趣的文章:- 詳解Linux patch命令參數(shù)及用法