位置: IT常识 - 正文

Linux环境下VI/VIM编辑文件时无权限保存的解决方法(普通用户)(linux vim)

编辑:rootadmin
在Linux环境下,如果直接使用VI/VIM命令编辑没有修改权限的文件时,保存的时候就会提示用户无法进行保存操作,这里给出几种解决方法,需要的朋友可以参考一下... 18-04-19

推荐整理分享Linux环境下VI/VIM编辑文件时无权限保存的解决方法(普通用户)(linux vim),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:linux vimtutor,linux vimtutor,linux中vim怎么运行,linux vimrc,linux vi vim,linux中vim怎么运行,linux中vim怎么运行,linux vi vim,内容如对您有帮助,希望把文章链接给更多的朋友!

在Linux环境下,如果直接使用VI/VIM命令编辑没有修改权限的文件时,保存的时候就会提示用户无法进行保存操作,一般的解决方法只能是关闭文件重新以sudo权限打开该文件编辑后再保存(前提是用户具有sudo权限)。其实,在VI/VIM模式下通过一些简单的命令,就能在不关闭当前文件的情况下达到保存文件的目的。

方法一

关于%! sudo tee % > /dev/null这条命令的说明如下

此命令是把当前文件(即%)作为stdin传给sudo tee命令来执行。

方法二

在Linux上工作的朋友很可能遇到过这样一种情况,当你用Vim编辑完一个文件时,运行:wq保存退出,突然蹦出一个错误:

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

这表明文件是只读的,按照提示,加上!强制保存::w!,结果又一个错误出现:

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

文件明明存在,为何提示无法打开?这错误又代表什么呢?查看文档: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 directoryor the file name is not valid.

原来是可能没有权限造成的。此时你才想起,这个文件需要root权限才能编辑,而当前登陆的只是普通用户,在编辑之前你忘了使用sudo来启动Vim,所以才保存失败。于是为了防止修改丢失,你只好先把它保存为另外一个临时文件temp-file-name,然后退出Vim,再运行sudo mv temp-file-name readonly-file-name覆盖原文件。

但这样操作过于繁琐。而且如果只是想暂存此文件,还需要接着修改,则希望保留Vim的工作状态,比如编辑历史,buffer状态等等,该怎么办?能不能在不退出Vim的情况下获得root权限来保存这个文件?

解决方案

答案是可以,执行这样一条命令即可:

:w !sudo tee %

接下来我们来分析这个命令为什么可以工作。首先查看文档:help :w,向下滚动一点可以看到:

*: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} isexecuted like with ":!{cmd}", any '!' is replaced withthe previous command |:!|.The default [range] for the ":w" command is the whole buffer (1,$)

把这个使用方法对应前面的命令,如下所示:

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

我们并未指定range,参见帮助文档最下面一行,当range未指定时,默认情况下是整个文件。此外,这里也没有指定opt。

Vim中执行外部命令

接下来是一个叹号!,它表示其后面部分是外部命令,即sudo tee %。文档中说的很清楚,这和直接执行:!{cmd}是一样的效果。后者的作用是打开shell执行一个命令,比如,运行:!ls,会显示当前工作目录下的所有文件,这非常有用,任何可以在shell中执行的命令都可以在不退出Vim的情况下运行,并且可以将结果读入到Vim中来。试想,如果你要在Vim中插入当前工作路径或者当前工作路径下的所有文件名,你可以运行:

:r !pwd或:r !ls

此时所有的内容便被读入至Vim,而不需要退出Vim,执行命令,然后拷贝粘贴至Vim中。有了它,Vim可以自由的操作shell而无需退出。

命令的另一种表示形式

再看前面的文档:

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

所以实际上这个:w并未真的保存当前文件,就像执行:w new-file-name时,它将当前文件的内容保存到另外一个new-file-name的文件中,在这里它相当于一个另存为,而不是保存。它将当前文档的内容写到后面cmd的标准输入中,再来执行cmd,所以整个命令可以转换为一个具有相同功能的普通shell命令:

$ cat readonly-file-name | sudo tee %

这样看起来”正常”些了。其中sudo很好理解,意为切换至root执行后面的命令,tee和%是什么呢?

%的意义

我们先来看%,执行:help cmdline-special可以看到:

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

在执行外部命令时,%会扩展成当前文件名,所以上述的cmd也就成了sudo tee readonly-file-name。此时整个命令即:

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

注意:在另外一个地方我们也经常用到%,没错,替换。但是那里%的作用不一样,执行:help :%查看文档:

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

Linux环境下VI/VIM编辑文件时无权限保存的解决方法(普通用户)(linux vim)

在替换中,%的意义是代表整个文件,而不是文件名。所以对于命令:%s/old/new/g,它表示的是替换整篇文档中的old为new,而不是把文件名中的old换成new。

tee的作用

现在只剩一个难点: tee。它究竟有何用?维基百科上对其有一个详细的解释,你也可以查看man page。下面这幅图很形象的展示了tee是如何工作的:

ls -l的输出经过管道传给了tee,后者做了两件事,首先拷贝一份数据到文件file.txt,同时再拷贝一份到其标准输出。数据再次经过管道传给less的标准输入,所以它在不影响原有管道的基础上对数据作了一份拷贝并保存到文件中。看上图中间部分,它很像大写的字母T,给数据流动增加了一个分支,tee的名字也由此而来。

现在上面的命令就容易理解了,tee将其标准输入中的内容写到了readonly-file-name中,从而达到了更新只读文件的目的。当然这里其实还有另外一半数据:tee的标准输出,但因为后面没有跟其它的命令,所以这份输出相当于被抛弃。当然也可以在后面补上> /dev/null,以显式的丢弃标准输出,但是这对整个操作没有影响,而且会增加输入的字符数,因此只需上述命令即可。

命令执行之后

运行完上述命令后,会出现下面的提示:

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

Vim提示文件更新,询问是确认还是重新加载文件。建议直接输入O,因为这样可以保留Vim的工作状态,比如编辑历史,buffer等,撤消等操作仍然可以继续。而如果选择L,文件会以全新的文件打开,所有的工作状态便丢失了,此时无法执行撤消,buffer中的内容也被清空。

更简单的方案:映射

上述方式非常完美的解决了文章开始提出的问题,但毕竟命令还是有些长,为了避免每次输入一长串的命令,可以将它映射为一个简单的命令加到.vimrc中:

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

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

这样,简单的运行:w!!即可。命令后半部分> /dev/null在前面已经解释过,作用为显式的丢掉标准输出的内容。

另一种思路

至此,一个比较完美但很tricky的方案已经完成。你可能会问,为什么不用下面这样更常见的命令呢?这不是更容易理解,更简单一些么?

:w !sudo cat > %

重定向的问题

我们来分析一遍,像前面一样,它可以被转换为相同功能的shell命令:

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

这条命令看起来一点问题没有,可一旦运行,又会出现另外一个错误:

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

shell returned 1

这是怎么回事?不是明明加了sudo么,为什么还提示说没有权限?稍安勿躁,原因在于重定向,它是由shell执行的,在一切命令开始之前,shell便会执行重定向操作,所以重定向并未受sudo影响,而当前的shell本身也是以普通用户身份启动,也没有权限写此文件,因此便有了上面的错误。

重定向方案

这里介绍了几种解决重定向无权限错误的方法,当然除了tee方案以外,还有一种比较方便的方案:以sudo打开一个shell,然后在该具有root权限的shell中执行含重定向的命令,如:

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

可是这样执行时,由于单引号的存在,所以在Vim中%并不会展开,它被原封不动的传给了shell,而在shell中,一个单独的%相当于nil,所以文件被重定向到了nil,所有内容丢失,保存文件失败。

既然是由于%没有展开导致的错误,那么试着将单引号'换成双引号"再试一次:

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

成功!这是因为在将命令传到shell去之前,%已经被扩展为当前的文件名。有关单引号和双引号的区别可以参考这里,简单的说就是单引号会将其内部的内容原封不动的传给命令,但是双引号会展开一些内容,比如变量,转义字符等。

当然,也可以像前面一样将它映射为一个简单的命令并添加到.vimrc中:

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

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

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

写在结尾

至此,借助Vim强大的灵活性,实现了两种方案,可以在以普通用户启动的Vim中保存需root权限的文件。两者的原理类似,都是利用了Vim可以执行外部命令这一特性,区别在于使用不同的shell命令。如果你还有其它的方案,欢迎给我留言。

(全文完)

feihu

2014.07.30 于 Shenzhen

本文链接地址:https://www.jiuchutong.com/zhishi/277931.html 转载请保留说明!

上一篇:苹果电脑双系统切换教程(苹果电脑双系统好不好)

下一篇:macOS big sur电池百分比显示怎么设置?(macos big sur 电池)

  • 联想打印机怎么连接手机(联想打印机怎么放纸)

    联想打印机怎么连接手机(联想打印机怎么放纸)

  • 陌陌注销必须审核14天吗(陌陌注销必须审核吗)

    陌陌注销必须审核14天吗(陌陌注销必须审核吗)

  • airpods打电话没有声音(airpods打电话没声音其它正常)

    airpods打电话没有声音(airpods打电话没声音其它正常)

  • 腾讯会员怎样设置多人(腾讯会员怎样设置账号密码)

    腾讯会员怎样设置多人(腾讯会员怎样设置账号密码)

  • 华为手机微信聊天记录怎么迁移(华为手机微信聊天记录删了能恢复吗)

    华为手机微信聊天记录怎么迁移(华为手机微信聊天记录删了能恢复吗)

  • 苹果11微信频繁闪退(苹果微信频繁闪退)

    苹果11微信频繁闪退(苹果微信频繁闪退)

  • 关机微信运动还计步吗(关机微信运动还变吗)

    关机微信运动还计步吗(关机微信运动还变吗)

  • 拼多多是怎么个拼法(拼多多怎么个人推流)

    拼多多是怎么个拼法(拼多多怎么个人推流)

  • 9v电池能充电吗(9v电池能用手机充电器充电吗)

    9v电池能充电吗(9v电池能用手机充电器充电吗)

  • 苹果11怎么关机充电(苹果11怎么关机不了)

    苹果11怎么关机充电(苹果11怎么关机不了)

  • iphone11pro max录屏怎么设置(iphone11pro max录屏在哪里)

    iphone11pro max录屏怎么设置(iphone11pro max录屏在哪里)

  • 小红书一直闪退怎么回事(小红书一直闪退打不开)

    小红书一直闪退怎么回事(小红书一直闪退打不开)

  • 手机系统更新不了怎么回事(手机系统更新不提醒设置)

    手机系统更新不了怎么回事(手机系统更新不提醒设置)

  • 苹果8的gps有快速开关吗(苹果8 gps在哪里开)

    苹果8的gps有快速开关吗(苹果8 gps在哪里开)

  • rm视频文件用什么打开(rm格式的视频文件用什么播放器)

    rm视频文件用什么打开(rm格式的视频文件用什么播放器)

  • ps怎么闭合路径载入选取(ps闭合路径后怎么建立选区)

    ps怎么闭合路径载入选取(ps闭合路径后怎么建立选区)

  • 华为屏幕上的圆圈怎样取消(华为屏幕上的圆点)

    华为屏幕上的圆圈怎样取消(华为屏幕上的圆点)

  • 苹果11怎么设置九键输入法(苹果11怎么设置来电闪光灯)

    苹果11怎么设置九键输入法(苹果11怎么设置来电闪光灯)

  • 米兔定位电话怎么解绑(米兔定位电话怎么使用教程)

    米兔定位电话怎么解绑(米兔定位电话怎么使用教程)

  • qq音乐交易额度限制怎么办(qq音乐交易额度限制1138)

    qq音乐交易额度限制怎么办(qq音乐交易额度限制1138)

  • 系统蓝屏怎么修复(系统蓝屏原因分析)

    系统蓝屏怎么修复(系统蓝屏原因分析)

  • word表格怎么求平均值(word表格怎么求和一行数据)

    word表格怎么求平均值(word表格怎么求和一行数据)

  • 如何删除金山文档小程序(如何删除金山文档中的文件QQ)

    如何删除金山文档小程序(如何删除金山文档中的文件QQ)

  • 7.0.4微信怎么改铃声(微信8.0.7怎么改微信号)

    7.0.4微信怎么改铃声(微信8.0.7怎么改微信号)

  • 台式电脑蓝牙在哪里(台式电脑蓝牙在哪里打开win7旗舰版)

    台式电脑蓝牙在哪里(台式电脑蓝牙在哪里打开win7旗舰版)

  • 通信方式有哪些

    通信方式有哪些

  • 为什么移动4g突然变成h了 (为什么移动4g突然变成e了)

    为什么移动4g突然变成h了 (为什么移动4g突然变成e了)

  • yumdownloader命令  从Yum存储库下载软件包(yum命令安装软件)

    yumdownloader命令 从Yum存储库下载软件包(yum命令安装软件)

  • 帝国cms模板文件存在哪个文件目录里(帝国cms模板文件在哪)

    帝国cms模板文件存在哪个文件目录里(帝国cms模板文件在哪)

  • 哪些土地房产转让合法
  • 补发上年度奖金如何计税
  • 红冲的发票能不能取消
  • 怎么从营业执照上看是不是一般纳税人
  • 进口商品仓库
  • 劳务费达到多少钱需要招标
  • 应税销售行为的购买方为消费者个人的可以开专票吗
  • 企业没有进项发票又开出很多发票
  • 企业年金也需要缴纳吗
  • 小型微利企业认定标准2023年
  • 职工退休金计入个人账户
  • 自己开发app能享受医保吗
  • 非股东可以投资项目吗
  • 一般纳税人净利润300万企业所得税怎么算
  • 公司筹建期间发生的费用没有发票
  • 非公司私营企业属于什么类型
  • 金税盘如何红冲已抵扣专票
  • 小企业会计准则和企业会计准则的区别
  • 公司房产税如何计算器
  • 公司购买的办公桌椅属于固定资产吗
  • 每月的话费到下个月会扣吗
  • 其他权益工具投资公允价值变动怎么计算
  • 小微企业可以享受六税两费吗
  • 搜索特定
  • pc端微信怎么更新
  • php获取网站内容
  • 埃姆雷莫尔
  • php获取数组的值
  • php保存远程图片
  • ajax向php传值
  • 页面访问升级出错怎么办
  • vue.js in action
  • 如何让header居中
  • linux用cat命令查看
  • python合并多个excel
  • 一般纳税人销售自己使用过的汽车
  • 转出未交增值税借方余额表示什么
  • mongodb起源
  • 织梦相关文章调用
  • 付国外专利费用需办什么手续
  • 定额计税的有哪些
  • sql server 查看
  • 注销公司流程超详细
  • 注册公司注册公司
  • 小规模纳税人减半征收的六税两费
  • 工会经费可以列支哪些项目
  • 什么是企业所得税收入
  • 基建和技改的区别
  • 公司给买员工保险
  • 人力资源社保代缴
  • 跨年发票应该如何处理
  • 货物逾期保管费怎么算
  • 应交税费包括哪些科目
  • 现金和银行存款的区别
  • 周转材料包装物的账务处理
  • 合并报表怎么合并
  • 服务佣金最高可以收多少
  • sqlserver数据库脚本
  • winxp系统提速
  • macbookzen
  • winex.exe - winex是什么进程
  • os x 10.10 yosemite自动纠正怎么关?os x yosemite自动纠正功能关闭教程
  • winxp回收站清空了怎么恢复
  • win10闹钟软件
  • PadExe.exe - PadExe是什么进程 有什么用
  • win7睡眠设置在哪
  • cocos2dx 2.2.2
  • mongoose怎么用
  • Cocos2d唯一死敌的崛起,OGEngine来了
  • perl命令行参数内建数组
  • html5+javascript实现简单上传的注意细节
  • js日历插件日期选择器
  • 不用jquery用什么
  • javascript面向对象 第三方类库
  • 税务软件服务商
  • 辽宁省国家税务总局
  • 建筑施工企业研发中心建设
  • 于开年 山东省国家税务局
  • 广东省电子税务局app下载手机版
  • 国家税务总局云平台网址
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设