溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Linux環(huán)境下VI/VIM編輯文件時(shí)無(wú)權(quán)限保存怎么辦

發(fā)布時(shí)間:2021-06-11 11:57:40 來(lái)源:億速云 閱讀:1089 作者:小新 欄目:系統(tǒng)運(yùn)維

這篇文章主要為大家展示了“Linux環(huán)境下VI/VIM編輯文件時(shí)無(wú)權(quán)限保存怎么辦”,內(nèi)容簡(jiǎn)而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“Linux環(huán)境下VI/VIM編輯文件時(shí)無(wú)權(quán)限保存怎么辦”這篇文章吧。

在Linux環(huán)境下,如果直接使用VI/VIM命令編輯沒(méi)有修改權(quán)限的文件時(shí),保存的時(shí)候就會(huì)提示用戶無(wú)法進(jìn)行保存操作,一般的解決方法只能是關(guān)閉文件重新以sudo權(quán)限打開該文件編輯后再保存(前提是用戶具有sudo權(quán)限)。其實(shí),在VI/VIM模式下通過(guò)一些簡(jiǎn)單的命令,就能在不關(guān)閉當(dāng)前文件的情況下達(dá)到保存文件的目的。

方法一

關(guān)于%! sudo tee % > /dev/null這條命令的說(shuō)明如下

此命令是把當(dāng)前文件(即%)作為stdin傳給sudo tee命令來(lái)執(zhí)行。

方法二
 

在Linux上工作的朋友很可能遇到過(guò)這樣一種情況,當(dāng)你用Vim編輯完一個(gè)文件時(shí),運(yùn)行:wq保存退出,突然蹦出一個(gè)錯(cuò)誤:

E45: 'readonly' option is set (add ! to override)

這表明文件是只讀的,按照提示,加上!強(qiáng)制保存::w!,結(jié)果又一個(gè)錯(cuò)誤出現(xiàn):

"readonly-file-name" E212: Can't open file for writing

文件明明存在,為何提示無(wú)法打開?這錯(cuò)誤又代表什么呢?查看文檔:help E212:

For some reason the file you are writing to cannot be created or overwritten.
The reason could be that you do not have permission to write in the directory
or the file name is not valid.

原來(lái)是可能沒(méi)有權(quán)限造成的。此時(shí)你才想起,這個(gè)文件需要root權(quán)限才能編輯,而當(dāng)前登陸的只是普通用戶,在編輯之前你忘了使用sudo來(lái)啟動(dòng)Vim,所以才保存失敗。于是為了防止修改丟失,你只好先把它保存為另外一個(gè)臨時(shí)文件temp-file-name,然后退出Vim,再運(yùn)行sudo mv temp-file-name readonly-file-name覆蓋原文件。

但這樣操作過(guò)于繁瑣。而且如果只是想暫存此文件,還需要接著修改,則希望保留Vim的工作狀態(tài),比如編輯歷史,buffer狀態(tài)等等,該怎么辦?能不能在不退出Vim的情況下獲得root權(quán)限來(lái)保存這個(gè)文件?

解決方案

答案是可以,執(zhí)行這樣一條命令即可:

:w !sudo tee %

接下來(lái)我們來(lái)分析這個(gè)命令為什么可以工作。首先查看文檔:help :w,向下滾動(dòng)一點(diǎn)可以看到:

	*:w_c* *:write_c*
:[range]w[rite] [++opt] !{cmd}
			Execute {cmd} with [range] lines as standard input
			(note the space in front of the '!').  {cmd} is
			executed like with ":!{cmd}", any '!' is replaced with
			the previous command |:!|.

The default [range] for the ":w" command is the whole buffer (1,$)

把這個(gè)使用方法對(duì)應(yīng)前面的命令,如下所示:

:       w               !sudo tee %
|       |               |  |
:[range]w[rite] [++opt] !{cmd}

我們并未指定range,參見幫助文檔最下面一行,當(dāng)range未指定時(shí),默認(rèn)情況下是整個(gè)文件。此外,這里也沒(méi)有指定opt。

Vim中執(zhí)行外部命令

接下來(lái)是一個(gè)嘆號(hào)!,它表示其后面部分是外部命令,即sudo tee %。文檔中說(shuō)的很清楚,這和直接執(zhí)行:!{cmd}是一樣的效果。后者的作用是打開shell執(zhí)行一個(gè)命令,比如,運(yùn)行:!ls,會(huì)顯示當(dāng)前工作目錄下的所有文件,這非常有用,任何可以在shell中執(zhí)行的命令都可以在不退出Vim的情況下運(yùn)行,并且可以將結(jié)果讀入到Vim中來(lái)。試想,如果你要在Vim中插入當(dāng)前工作路徑或者當(dāng)前工作路徑下的所有文件名,你可以運(yùn)行:

:r !pwd或:r !ls

此時(shí)所有的內(nèi)容便被讀入至Vim,而不需要退出Vim,執(zhí)行命令,然后拷貝粘貼至Vim中。有了它,Vim可以自由的操作shell而無(wú)需退出。

命令的另一種表示形式

再看前面的文檔:

Execute {cmd} with [range] lines as standard input

所以實(shí)際上這個(gè):w并未真的保存當(dāng)前文件,就像執(zhí)行:w new-file-name時(shí),它將當(dāng)前文件的內(nèi)容保存到另外一個(gè)new-file-name的文件中,在這里它相當(dāng)于一個(gè)另存為,而不是保存。它將當(dāng)前文檔的內(nèi)容寫到后面cmd的標(biāo)準(zhǔn)輸入中,再來(lái)執(zhí)行cmd,所以整個(gè)命令可以轉(zhuǎn)換為一個(gè)具有相同功能的普通shell命令:

$ cat readonly-file-name | sudo tee %

這樣看起來(lái)”正?!毙┝?。其中sudo很好理解,意為切換至root執(zhí)行后面的命令,tee和%是什么呢?

%的意義

我們先來(lái)看%,執(zhí)行:help cmdline-special可以看到:

In Ex commands, at places where a file name can be used, the following
characters have a special meaning. These can also be used in the expression
function expand() |expand()|.
% Is replaced with the current file name. *:_%* *c_%*

在執(zhí)行外部命令時(shí),%會(huì)擴(kuò)展成當(dāng)前文件名,所以上述的cmd也就成了sudo tee readonly-file-name。此時(shí)整個(gè)命令即:

$ cat readonly-file-name | sudo tee readonly-file-name

注意:在另外一個(gè)地方我們也經(jīng)常用到%,沒(méi)錯(cuò),替換。但是那里%的作用不一樣,執(zhí)行:help :%查看文檔:

Line numbers may be specified with: *:range* *E14* *{address}*
{number} an absolute line number
...
% equal to 1,$ (the entire file) *:%*

在替換中,%的意義是代表整個(gè)文件,而不是文件名。所以對(duì)于命令:%s/old/new/g,它表示的是替換整篇文檔中的old為new,而不是把文件名中的old換成new。

tee的作用

現(xiàn)在只剩一個(gè)難點(diǎn): tee。它究竟有何用?維基百科上對(duì)其有一個(gè)詳細(xì)的解釋,你也可以查看man page。下面這幅圖很形象的展示了tee是如何工作的:

Linux環(huán)境下VI/VIM編輯文件時(shí)無(wú)權(quán)限保存怎么辦

ls -l的輸出經(jīng)過(guò)管道傳給了tee,后者做了兩件事,首先拷貝一份數(shù)據(jù)到文件file.txt,同時(shí)再拷貝一份到其標(biāo)準(zhǔn)輸出。數(shù)據(jù)再次經(jīng)過(guò)管道傳給less的標(biāo)準(zhǔn)輸入,所以它在不影響原有管道的基礎(chǔ)上對(duì)數(shù)據(jù)作了一份拷貝并保存到文件中。看上圖中間部分,它很像大寫的字母T,給數(shù)據(jù)流動(dòng)增加了一個(gè)分支,tee的名字也由此而來(lái)。

現(xiàn)在上面的命令就容易理解了,tee將其標(biāo)準(zhǔn)輸入中的內(nèi)容寫到了readonly-file-name中,從而達(dá)到了更新只讀文件的目的。當(dāng)然這里其實(shí)還有另外一半數(shù)據(jù):tee的標(biāo)準(zhǔn)輸出,但因?yàn)楹竺鏇](méi)有跟其它的命令,所以這份輸出相當(dāng)于被拋棄。當(dāng)然也可以在后面補(bǔ)上> /dev/null,以顯式的丟棄標(biāo)準(zhǔn)輸出,但是這對(duì)整個(gè)操作沒(méi)有影響,而且會(huì)增加輸入的字符數(shù),因此只需上述命令即可。

命令執(zhí)行之后

運(yùn)行完上述命令后,會(huì)出現(xiàn)下面的提示:

W12: Warning: File "readonly-file-name" has changed and the buffer was changed in Vim as well
See ":help W12" for more info.
[O]K, (L)oad File:

Vim提示文件更新,詢問(wèn)是確認(rèn)還是重新加載文件。建議直接輸入O,因?yàn)檫@樣可以保留Vim的工作狀態(tài),比如編輯歷史,buffer等,撤消等操作仍然可以繼續(xù)。而如果選擇L,文件會(huì)以全新的文件打開,所有的工作狀態(tài)便丟失了,此時(shí)無(wú)法執(zhí)行撤消,buffer中的內(nèi)容也被清空。

更簡(jiǎn)單的方案:映射

上述方式非常完美的解決了文章開始提出的問(wèn)題,但畢竟命令還是有些長(zhǎng),為了避免每次輸入一長(zhǎng)串的命令,可以將它映射為一個(gè)簡(jiǎn)單的命令加到.vimrc中:

" Allow saving of files as sudo when I forgot to start vim using sudo.

cmap w!! w !sudo tee > /dev/null %

這樣,簡(jiǎn)單的運(yùn)行:w!!即可。命令后半部分> /dev/null在前面已經(jīng)解釋過(guò),作用為顯式的丟掉標(biāo)準(zhǔn)輸出的內(nèi)容。

另一種思路

至此,一個(gè)比較完美但很tricky的方案已經(jīng)完成。你可能會(huì)問(wèn),為什么不用下面這樣更常見的命令呢?這不是更容易理解,更簡(jiǎn)單一些么?

:w !sudo cat > %

重定向的問(wèn)題

我們來(lái)分析一遍,像前面一樣,它可以被轉(zhuǎn)換為相同功能的shell命令:

$ cat readonly-file-name | sudo cat > %

這條命令看起來(lái)一點(diǎn)問(wèn)題沒(méi)有,可一旦運(yùn)行,又會(huì)出現(xiàn)另外一個(gè)錯(cuò)誤:

/bin/sh: readonly-file-name: Permission denied

shell returned 1

這是怎么回事?不是明明加了sudo么,為什么還提示說(shuō)沒(méi)有權(quán)限?稍安勿躁,原因在于重定向,它是由shell執(zhí)行的,在一切命令開始之前,shell便會(huì)執(zhí)行重定向操作,所以重定向并未受sudo影響,而當(dāng)前的shell本身也是以普通用戶身份啟動(dòng),也沒(méi)有權(quán)限寫此文件,因此便有了上面的錯(cuò)誤。

重定向方案

這里介紹了幾種解決重定向無(wú)權(quán)限錯(cuò)誤的方法,當(dāng)然除了tee方案以外,還有一種比較方便的方案:以sudo打開一個(gè)shell,然后在該具有root權(quán)限的shell中執(zhí)行含重定向的命令,如:

:w !sudo sh -c 'cat > %'

可是這樣執(zhí)行時(shí),由于單引號(hào)的存在,所以在Vim中%并不會(huì)展開,它被原封不動(dòng)的傳給了shell,而在shell中,一個(gè)單獨(dú)的%相當(dāng)于nil,所以文件被重定向到了nil,所有內(nèi)容丟失,保存文件失敗。

既然是由于%沒(méi)有展開導(dǎo)致的錯(cuò)誤,那么試著將單引號(hào)'換成雙引號(hào)"再試一次:

:w !sudo sh -c "cat > %"

成功!這是因?yàn)樵趯⒚顐鞯絪hell去之前,%已經(jīng)被擴(kuò)展為當(dāng)前的文件名。有關(guān)單引號(hào)和雙引號(hào)的區(qū)別可以參考這里,簡(jiǎn)單的說(shuō)就是單引號(hào)會(huì)將其內(nèi)部的內(nèi)容原封不動(dòng)的傳給命令,但是雙引號(hào)會(huì)展開一些內(nèi)容,比如變量,轉(zhuǎn)義字符等。

當(dāng)然,也可以像前面一樣將它映射為一個(gè)簡(jiǎn)單的命令并添加到.vimrc中:

" Allow saving of files as sudo when I forgot to start vim using sudo.

cmap w!! w !sudo sh -c "cat > %"

注意:這里不再需要把輸出重定向到/dev/null中。

以上是“Linux環(huán)境下VI/VIM編輯文件時(shí)無(wú)權(quán)限保存怎么辦”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI