Welcome 微信登录

首页 / 操作系统 / Linux / vi/vim高级命令集粹

1.交换两个字符位置xp2.上下两行调换ddp3.把文件内容反转:g/^/m0/ (未通过)4.上下两行合并J5.删除所有行dG6.从当前位置删除到行尾d$7.从当前位置复制到行尾y$ 如果要粘贴到其他地方 p 就可以了由于vi 是建立在 EX 上的 所以 当键入 : 时就来到了 EX 命令状态8.:ab string strings例如 ":ab usa United States of America" ,当你在文见里插入 usa 时United States of America 就蹦出来了9.:map keys new_seq定义你当前 键盘命令10.:set [all]vi or ex 的编辑状态如 显示每行 :set nu11.在命令状态下,nyy表示拷贝从光标行起的下n行内容,p表示paste,可刚复制的内容粘贴在光标处的下面。12.单个字符替换用r,覆盖多个字符用R,用多个字符替换一个字符用s,整行替换用S13.:%s/old_word/new_word/g这个指令是于在整个文件中替换特定字符串14.光标控制k:上移 nk 上移n行j:下移 nj 下移n行将光标移到第n行,按下 mk将光标移到第m行,按下 "ay"k即将第n到m的行存到a寄存器,以此类推,b,c........寄存器等这样就可以将你常用的需要复用的内容粘贴到不同的寄存器中以备用想粘贴到某处,直接将光标移到某地,按下 "ap 即可,以此类推,b,c........寄存器等在当前屏幕中H 跳到第一行M 跳到中间一行L 跳到最后一行15.表8-2 删除命令删除命令操作d l 删除当前字符(与x命令功能相同)d 0 删除到某一行的开始位置d ^ 删除到某一行的第一个字符位置(不包括空格或TA B字符)d w 删除到某个单词的结尾位置d 3 w 删除到第三个单词的结尾位置d b 删除到某个单词的开始位置d W 删除到某个以空格作为分隔符的单词的结尾位置d B 删除到某个以空格作为分隔符的单词的开始位置d 7 B 删除到前面7个以空格作为分隔符的单词的开始位置d) 删除到某个语句的结尾位置d 4) 删除到第四个语句的结尾位置d( 删除到某个语句的开始位置d } 删除到某个段落的结尾位置d { 删除到某个段落的开始位置d 7 { 删除到当前段落起始位置之前的第7个段落位置d d 删除当前行d /t e x t 删除从文本中出现" t e x t"中所指定字样的位置,一直向前直到下一个该字样所出现的位置(但不包括该字样)之间的内容d fc 删除从文本中出现字符"c"的位置,一直向前直到下一个该字符所出现的位置(包括该字符)之间的内容d tc 删除当前行直到下一个字符" c"所出现位置之间的内容D 删除到某一行的结尾d $ 删除到某一行的结尾5 d d 删除从当前行所开始的5行内容d L 删除直到屏幕上最后一行的内容d H 删除直到屏幕上第一行的内容d G 删除直到工作缓存区结尾的内容d 1 G 删除直到工作缓存区开始的内容修改命令操作c l 更改当前字符c w 修改到某个单词的结尾位置c 3 w 修改到第三个单词的结尾位置c b 修改到某个单词的开始位置c W 修改到某个以空格作为分隔符的单词的结尾位置c B 修改到某个以空格作为分隔符的单词的开始位置c 7 B 修改到前面7个以空格作为分隔符的单词的开始位置c 0 修改到某行的结尾位置c) 修改到某个语句的结尾位置c 4) 修改到第四个语句的结尾位置c( 修改到某个语句的开始位置c } 修改到某个段落的结尾位置c { 修改到某个段落的开始位置c 7 { 修改到当前段落起始位置之前的第7个段落位置c tc 修改当前行直到下一个字符c所出现位置之间的内容C 修改到某一行的结尾c c 修改当前行5 c c 修改从当前行所开始的5行内容.重复上一次修改!表8-4 替换命令替换命令操作s 将当前字符替换为一个或多个字符S 将当前行替换为一个或多个字符5 s 将从当前字符开始的5个字符替换为一个或多个字符vi替换使用规则::g/s1/s/s2/s3/g第一个g表示对每一个包括s1的行都进行替换,第二个g表示对每一行包括s1的行所有的s2都用s3替换s表示替换,s2是要被替换的字符串,他可以和s1相同(如果相同的话用//代替),s3是替换字符串16.fx往右移动到 x 字符上Fx往左移动到 x 字符上tx往右移动到 x 字符前Tx往左移动到 x 字符后(注意:以上四个命令中,其中x是键入的字符);分号,配合 f 和 t 使用,重复一次,逗号,配合 f 和 t 使用,反方向重复一次17. vi 环境选项 Solaris kshnoautoindent nomodelines noshowmodeautoprint nonumber noslowopennoautowrite nonovice tabstop=8nobeautify nooptimize taglength=0directory=/var/tmp paragraphs=IPLPPPQPP LIpplpipnpbtags=tags /usr/lib/tagsnoedcompatible prompt tagstacknoerrorbells noreadonly term=vt100noexrc redraw noterseflash remap timeouthardtabs=8 report=5 ttytype=vt100noignorecase scroll=11 warnnolisp sections=NHSHH HUuhsh+c window=23nolist shell=/bin/ksh wrapscanmagic shiftwidth=8 wrapmargin=0mesg noshowmatch nowriteanyFor C-Shell:setenv EXINIT "set nu"For Bourne or Korn Shell:EXINIT="set nu"; export EXINITFor Korn Shell Only (alternate method):typeset -x EXINIT="set nu"在 .profile 里设置 vi 的环境选项 , 以上均测试过18.标记文本mchar 用字母char标记当前光标的位置`char 移至char所标记处"char 移至char标记所在行的开头处" 移至当前行上一次所在位置(在光标移动之后)――一个双引号"" 移至当前行上第一次所在位置的行的开头处(在光标移动之后)――两个单引号19.同时vi多个文件时,CTRL-SHIFT-6回到上一个文件,在本次vi的文件和上次vi的文件之间切换。但是我发现一个BUG:在用CTRL-SHIFT-6切换到上一个文件后,用:args查看多文件vi状态时,屏幕底部仍然显示目前vi的是刚才的文件。(在HP-UX,Solaris,AIX上通过)也可以使用::e#进行切换20.sco 下VI 要在文本前同样的字符加用%s/^/要加的内容/g 要在文本后同样的字符加%s/$/要加的内容/g21.如何去掉文本中的 ^M 硬回车?不必用binary传回去再ascii传回来的方式,用shell或者unix语句实现。cat filename |tr -d "15" >newfile不同的unix系统还存在一些其他不同的命令,如:doscpsed 也可以实现这个功能.dos2unix filename filename2反之unix2dos filename filename2在vi 中用:$s/^M//g^是crtl-V crtl-M22.如何在"unix命令行"下将一个文件的某字符串用另一个串换掉sed "s/string1/string2/gp" file1 > file223.将/etc/hosts下所有的地址都ping 2次1 #/usr/bin/sh2 #grad /etc/hosts and ping each address3 cat /etc/hosts|grep -v "^#" | while read LINE4 do5 ADDR=`awk "{print $1}"`6 for MACHINE in $ADDR7 do8 ping $MACHINE -n 29 done10 done24到前一个函数[[ ,到下一个函数]] ,括号配对% ,交叉参考Ctrl_] (事先用ctags做索引),回来用e# ` 编辑一个函数:vi -t 函数名 ,编辑加密文本vi -X25在插入模式下ctrl+p,自动补齐剩余单词,以赖规则:tags,以有的单词等等---------------------------------------------------------------------------当今世界,文本编辑器种类繁多,大有"乱花渐欲迷人眼"之势。中国有句古语:手巧不如家什妙,作为IT业的专业人士,选择一款优秀的编辑软件至关重要。笔者认为:LINUX下的VIM※以其强大的功能和无穷的魅力将使您终生受益。作者:闫石 (iloveibm@163.com)来自:http://www-900.ibm.com/developerWork...15/index.shtml由于被广泛移植,无论是PC机的DOS和WINDOWS,还是RISC/6000的AIX,乃至于IBM的大型机S/390,都能见到VIM的身影。然而,对于初学者,VIM的用户界面与使用方法非常不符合常规,甚至认为它比较混乱,无所适从。事实上,VIM编辑器是专门为经验丰富的用户设计的,它的界面和使用方法提供了更快的速度和更强的功能。对于熟知它的用户,VIM的许多特性节省了时间和击键次数,并可以完成一些其他编辑器无法完成的功能。学习的最好方法是实践,唯有如此,才能真正掌握其中的精髓。文中列举的实例,都是笔者在实际工作中遇到的,具有一定的代表性,请大家在阅读的过程中仔细体会。好了,现在让我们共同畅游神奇的VIM的世界!例一、两个常用的指令序列xp 左右交换光标处两字符的位置。ddp 上下交换光标处两行的位置。例二、重复输入同一字符有时,我们可能想多次输入同一字符,VIM的插入功能可以很好的完成这项工作命令 80i=^ESC 一次可以输入80个字符= ,当然,80a=^ESC 也可以完成上述功能。请注意:此处的^ESC表示键盘左上方上的ESC键。例三、将两个文本数据文件按行逐条合并,并给出标尺数据文件1内容如下:1-----2-----3-----数据文件2内容如下:1=====2=====3=====要求的结果如下:|--------1---------2---------3---------4---------51-----1=====|--------1---------2---------3---------4---------52-----2=====|--------1---------2---------3---------4---------53-----3=====也许您会说,这还不简单,无非是反复拷贝、粘贴,任何一款文本编辑器都能完成上述功能。可是,如果这两个文件都很大,每个文件都成千上万行,恐怕简单的拷贝、粘贴就难以胜任了。因此,我们所关心的,是找到一种行之有效的方法,把枯燥乏味的工作留给计算机,我们只需发布指令。为达到此目的,请按以下步骤执行:㈠、将两文件合并,结果如下1-----2-----3-----1=====2=====3=====㈡、在两文件头尾相接的地方插入标志行,用以区分两个文件,本文采用的是一整行!字符1-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!1=====2=====3=====㈢、在标志行的下方输入标尺1-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====㈣、执行宏命令脚本merge_2r.vim,即在VIM编辑器中按如下键 :so merge_2r.vim 回车㈤、按下键盘上的=键,执行的结果如下|--------1---------2---------3---------4---------51-----1=====|--------1---------2---------3---------4---------52-----2=====|--------1---------2---------3---------4---------53-----3=====|--------1---------2---------3---------4---------5!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------5㈥、将最后三行删除,即可得到我们需要的结果|--------1---------2---------3---------4---------51-----1=====|--------1---------2---------3---------4---------52-----2=====|--------1---------2---------3---------4---------53-----3=====怎么样,简单吗?请大家自己实际尝试一下。下面,我来详细讲解宏命令脚本merge_2r.vim 。该脚本内容如下:"--------------------------------------------------------------------"Macro Function : Merge File1 And File2,Have Ruler in every record" Date : 2001/12/01" Author : Yan Shi"--------------------------------------------------------------------"1-----"2----- } Sample File1"3-----"!!!!!!!!!!!!!!!!!!!!!!!! Flag Row"|--------1---------2---------3---------4---------5 Ruler"1====="2===== } Sample File2"3====="--------------------------------------------------------------------:1:map = ma/!!!!!^M+:.co "a-1^M/!!!!!^M2+:.m"a^M+=前14行每行都以"开始,表明该行是注释行,实际并不执行,只是方便读者阅读,只有最后两行才是真正的代码行。请注意:本例中的^M表示键盘上的回车键,并非^和M两个字符。为了讲述清楚,我把命令行分解开,逐一说明。首先将第一行置为当前行,然后执行map命令,将一大串VIM指令映像给字符=。这一大串VIM指令共分9步执行:ma 将数据文件一的第一行标记为a1-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====/!!!!!^M 找到标志行,置为当前行1-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====+ 光标下移一行,即把标尺行置为当前行1-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====:.co "a-1^M 把标尺行复制到标记行(数据文件一的第一行)的上方|--------1---------2---------3---------4---------51-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====/!!!!!^M 再次找到标志行,置为当前行|--------1---------2---------3---------4---------51-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====2+ 光标下移2行,即数据文件二的第一行置为当前行|--------1---------2---------3---------4---------51-----2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------51=====2=====3=====:.m"a^M 把数据文件二的第一行移至标记行的下方|--------1---------2---------3---------4---------51-----1=====2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------52=====3=====+ 光标下移一行,即数据文件一的第二行置为当前行|--------1---------2---------3---------4---------51-----1=====2-----3-----!!!!!!!!!!!!!!!!!!!!!!!!|--------1---------2---------3---------4---------52=====3====== 这一步很关键,是典型的递归调用,重复完成以上步骤例四、在文件中置入行号工作中,我们有时希望把行号置入文件中,而VIM提供的功能 :set nu 只能显示行号,不能编辑或将其置入文件当中,下面的宏命令脚本row_num.vim可以完成此项功能。"------------------------------------------"Macro Function : Source File Add Row_Num" Date : 2001/12/01" Author : Yan Shi"------------------------------------------:%s/^/^I/:$:let end=line("."):1"------------------------------------------:let num=1:while num<=end:let line=getline("."):let temp=substitute(line,$,num,""):call setline(".",temp):+:let num=num+1:endwhile"------------------------------------------请注意:本例中的^I表示键盘上的TAB键,并非^和I两个字符。下面,我针对该宏命令脚本逐一讲解。:%s/^/^I/ 每一行的行首添加一个TAB字符:$ 到文件的末行:let end=line(".") 末行的行号 ==〉变量 END,函数line的功能是取得指定行的行号,此处参数"."表示当前行:1 到文件的首行"------------------------------------------:let num=1 1 ==〉计数器:while num<=end:let line=getline(".") 取当前行的内容 ==〉变量 LINE:let line=substitute(line,$,num,"") 在变量 LINE 的前面置入行号:call setline(".",line) 将变量 LINE 的内容写回当前行:+ 下移一行:let num=num+1 计数器加一:endwhile 循环执行,直到文件结束"------------------------------------------有关正则表达式的使用UNIX/LINUX下的很多工具之所以强大、灵活,关键是因为有了正则文法和元字符,这也是VIM乃至UNIX/LINUX系统的精华所在。正因为使用灵活,因此掌握起来比较吃力,如果不是真正理解,实际运用中会出现千奇百怪的错误。因此,有必要对这部分知识多花些气力。下面结合具体实例讲解。例五、有一文件,包含某外企的中国员工的资料,首先是姓名,然后是两个空格,其次是15位身份证号码。zhang.fei 430759701022003diao.chan 651302801225012guan.yu 342869680413001xi.shi 120638780214006liu.bei 210324650708001现在,有以下问题需要解决:按照外国人的习惯,应该是名在前,姓在后。因此,文件中的姓名字段需要修改。姓与名的首字母应该大写。根据身份证号码,还可以判断出生年月日,将其作为一个新字段添加。根据身份证号码,可以判断出性别。若为男性,添加male,若为女性,添加female 。将男女员工分开,男员工在前,女员工在后。将各字段数据左对齐最终结果如下:Fei.Zhang 430759701022003 1970/10/22 maleYu.Guan 342869680413001 1968/04/13 maleBei.Liu 210324650708001 1965/07/08 male-----------------------------------------------Chan.Diao 651302801225012 1980/12/25 femaleShi.Xi 120638780214006 1978/02/14 female为了完成上述功能,只需执行脚本employee.vim ,使用方法为 :so employee.vim 回车即可。脚本内容如下::%s/ / /:%s/(............)( *)/1/:%s/([A-Za-z][A-Za-z]*)(.)([A-Za-z][A-Za-z]*)/u32u1/:%s/$/ xxxxxx/:%s/([0-9]{6})([0-9]{6})([0-9]{3}) (xxxxxx)/123 2/:%s/(..)(..)(..)$/191/2/3:%s/$/ xxxxxx/:%s/([0-9]{14}[13579])(.*)(xxxxxx)/12male /:%s/([0-9]{14}[02468])(.*)(xxxxxx)/12female/:$:s/.*/&^M-----------------------------------------------:g/female/.m$在这个脚本中,使用了大量的正则表达式,这里仅对涉及到的正则表达式做一简要介绍。有关正则表达式的内容相当多,本文不可能占用大量篇幅叙述,请大家谅解。% 地址范围符号,代表文件中的所有行,作用等同于地址范围 1,$. 与任意单字符(换行符除外)匹配,例如 y.s 可以匹配 yas y.s 或 y s 等等。* 与前一字符的0次或多次出现匹配,例如 y*s 可以匹配 yys yyyyys 或 s 等等。$ 与行尾匹配。& 代表模式匹配中出现的字符串,例如 s/abc/&def 是把当前行的abc替换成abcdef 。[] 匹配[]中出现的字符,例如[abc]匹配字符 a,b 或 c ,[a-zA-Z]匹配所有的英文字符。( ) (和)之间出现的内容可以由 um来替代。123 替代(和)之间出现的内容。u 将后续字符串的首字母大写。{num} 与前一字符的num次出现匹配。现在,我们对脚本逐条讲解,希望能帮助大家理解正则文法。⑴:%s/ / /将文件中每行出现的2个空格替换为10个空格。zhang.fei 430759701022003diao.chan 651302801225012guan.yu 342869680413001xi.shi 120638780214006liu.bei 210324650708001⑵:%s/(............)( *)/1/保留行首的12个字符,将其余的空格删除,这样,前两个字段就对齐了。zhang.fei 430759701022003diao.chan 651302801225012guan.yu 342869680413001xi.shi 120638780214006liu.bei 210324650708001⑶:%s/([A-Za-z][A-Za-z]*)(.)([A-Za-z][A-Za-z]*)/u32u1/将文件中每行出现的雇员姓名互换,并将首字母大写。Fei.Zhang 430759701022003Chan.Diao 651302801225012Yu.Guan 342869680413001Shi.Xi 120638780214006Bei.Liu 210324650708001⑷:%s/$/ xxxxxx/在每一行的行尾添加2个空格和6个xFei.Zhang 430759701022003 xxxxxxChan.Diao 651302801225012 xxxxxxYu.Guan 342869680413001 xxxxxxShi.Xi 120638780214006 xxxxxxBei.Liu 210324650708001 xxxxxx⑸:%s/([0-9]{6})([0-9]{6})([0-9]{3}) (xxxxxx)/123 2/将xxxxxx替换成出生年月日。Fei.Zhang 430759701022003 701022Chan.Diao 651302801225012 801225Yu.Guan 342869680413001 680413Shi.Xi 120638780214006 780214Bei.Liu 210324650708001 650708⑹:%s/(..)(..)(..)$/191/2/3将年月日用/字符分隔,并在年前添加19。Fei.Zhang 430759701022003 1970/10/22Chan.Diao 651302801225012 1980/12/25Yu.Guan 342869680413001 1968/04/13Shi.Xi 120638780214006 1978/02/14Bei.Liu 210324650708001 1965/07/08⑺:%s/$/ xxxxxx/在每一行的行尾添加2个空格和6个xFei.Zhang 430759701022003 1970/10/22 xxxxxxChan.Diao 651302801225012 1980/12/25 xxxxxxYu.Guan 342869680413001 1968/04/13 xxxxxxShi.Xi 120638780214006 1978/02/14 xxxxxxBei.Liu 210324650708001 1965/07/08 xxxxxx⑻:%s/([0-9]{14}[13579])(.*)(xxxxxx)/12male /身份证号码末位是奇数的,将xxxxxx替换成maleFei.Zhang 430759701022003 1970/10/22 maleChan.Diao 651302801225012 1980/12/25 xxxxxxYu.Guan 342869680413001 1968/04/13 maleShi.Xi 120638780214006 1978/02/14 xxxxxxBei.Liu 210324650708001 1965/07/08 male⑼:%s/([0-9]{14}[02468])(.*)(xxxxxx)/12female/身份证号码末位是偶数的,将xxxxxx替换成femaleFei.Zhang 430759701022003 1970/10/22 maleChan.Diao 651302801225012 1980/12/25 femaleYu.Guan 342869680413001 1968/04/13 maleShi.Xi 120638780214006 1978/02/14 femaleBei.Liu 210324650708001 1965/07/08 male⑽:$ 到文件的最后一行⑾:s/.*/&^M-----------------------------------------------在文件的最末行插入一行 "-" 字符。Fei.Zhang 430759701022003 1970/10/22 maleChan.Diao 651302801225012 1980/12/25 femaleYu.Guan 342869680413001 1968/04/13 maleShi.Xi 120638780214006 1978/02/14 femaleBei.Liu 210324650708001 1965/07/08 male-----------------------------------------------⑿:g/female/.m$将所有的女员工记录移至文件尾。Fei.Zhang 430759701022003 1970/10/22 maleYu.Guan 342869680413001 1968/04/13 maleBei.Liu 210324650708001 1965/07/08 male-----------------------------------------------Chan.Diao 651302801225012 1980/12/25 femaleShi.Xi 120638780214006 1978/02/14 female笔者目前正在为某外资公司从事大型机(IBM S/390)的软件开发,一切工作都在TSO环境中进行。为了对编写的程序进行测试,必须准备测试数据。有过大型机开发经验的人会知道,通过TSO,输入字符型数据还可以,如果要输入16进制数据,操作起来很麻烦。因为16进制数是纵向排列的,输入时既不方便,又很容易错位。怎么解决呢?我尝试了几种办法,实际证明,用VIM最方便。例六、下列数据 1234567890ABCDEF ,将其变成 13579ACE 24680BDF 的形式,这样,数据就可以很方便的粘贴到TSO环境中了。下面给出宏命令脚本change_d.vim"----------------------------------------------------"Macro Function : Convert Char Arrange Direction"" Sample : 40 50 60 ==> 4 5 6" 0 0 0" Date : 2001/12/01" Author : Yan Shi"----------------------------------------------------:s/.*/&^M/:1:map = malx+$p-`al=说明如下:⑴ :s/.*/&^M/ 在数据行下方添加一空行。⑵ :1 回到文件的首行的首字符。⑶ :map = malx+$p-`al= 将一大串VIM命令映像给字符=① ma 将首字符标记为a② l 光标右移一个字符③ x 删除光标处字符④ + 移至下一行⑤ $ 到行尾⑥ p 将删除的字符粘贴⑦ - 回至上一行⑧ `a 返回到标记字符处⑨ l 光标右移一个字符⑩ = 递归调用,重复以上步骤,直到将该行所有的数据处理完。上面的这几个实例,展示了VIM强大的文本处理功能,但这远不能覆盖其全貌。VIM的命令很多,而且使用灵活,需要狠下一番气力才能熟练掌握。笔者年龄尚小,经验还很欠缺,希望本文能够起到抛砖引玉的作用。由于时间的原因,上述实例仅在DOS和WINDOWS环境下测试,没有在其他系统下进行进一步的测试,希望各位同行、前辈不吝赐教,谢谢!※ VIM 意为 VI Improved ,与VI99%向下兼容。而且,VIM提供了许多VI不具备的功能,内置了诸多函数,因此???建议有经验的VI用户对VIM有所了解,您会发现,转向VIM是明智之举。欲查询有关VIM的资料,请参考 http://www.vim.org注:本文使用 VIM 6.0 版本