上次我们用宏操作的方式在VIM中由一行代码批量生成了一系列相似的代码,原文在这里
这次我们用正则表达式的方式实现批量替换
把上次的小练习先贴出来
在GVIM下,将下面这张图的内容
改成下面这样
并且指出,要用批量操作的方式,不能一行一行的键入
在给出实现方法之前,先给出VIM中各种符号的意义
普通字符 | 含义 |
---|---|
. | 匹配任意一个字符 |
abc | 匹配方括号中的任意一个字符,可以使用 - 表示范围比如a-z0-9匹配小写字母和数字 |
^abc | 匹配除方括号中字符以外的字符 |
d | 匹配阿拉伯数字,等同于0~9 |
D | 匹配阿拉伯数字之外的任意字符,等同于^0~9 |
x | 匹配十六进制数字,等同于0-9A-Fa-f |
X | 匹配十六进制数字之外的任意字符,等同于 ^0-9A-Fa-f |
w | 匹配单词字母 |
W | 匹配单词字母之外的任意字符 |
t | 匹配Tab字符 |
s | 匹配空白符,等同于t |
S | 匹配非空白符,等同于 ^t |
特殊字符 | 含义 |
---|---|
* | 匹配 * 字符 |
. | 匹配 . 字符 |
/ | 匹配 / 字符 |
| 匹配 字符 |
[ | 匹配 [ 字符 |
表示数量的元字符 | 含义 |
---|---|
匹配0~任意个 | |
| 匹配1~任意个 |
? | 匹配0~1个 |
{n,m} | 匹配n ~ m个 |
{n} | 匹配 n 个 |
{n,} | 匹配 n ~ 任意个 |
{,m} | 匹配 0 ~ m 个 |
表示位置元字符 | 含义 |
---|---|
$ | 匹配行尾 |
^ | 匹配行首 |
< | 匹配单词词首 |
> | 匹配单词词尾 |
现在回到那个小练习,由于我们主要用正则来进行替换,因此我们先将第一行内容进行复制粘贴
代码语言:perl复制在普通模式下
gg //回到首行
yy //复制首行内容
19p //粘贴19次
之后变成这样了
再进行替换之前,介绍下VIM替换的方式
替换的格式为
代码语言:perl复制:s/a/b/g //将a替换成b
//s是substitute,g是global
我们先gg回到首行,然后看看加上g和不加g的区别
命令行执行如下内容
代码语言:perl复制:s/1/aa/
可以看到,只有第一行的第一个1
变成了aa
再执行如下内容
代码语言:perl复制u //undo,撤销上一步操作
:s/1/aa/g
可以看到此时首行所有的1
都被替换成了aa
虽然加上g
将首行的1
全部替换成了aa
,但也只有首行被替换了,如果多行都需要替换,我们不可能在每一行都执行这个操作,因此引入了%
我们执行下面的内容看看是什么结果
代码语言:perl复制u //撤销上一步操作
:%s/1/aa/
可以看到此时每行的第一个1
都被替换成了aa
如果我们只想替换其中几行的内容,可以将%
换成范围
:5,8s/1/aa/
此时只有5-8行的1
被替换了
由于我们这个小练习被替换成的内容不是固定的,因此需要再介绍一个函数
代码语言:perl复制=line(".")
这个函数是获取行号的意思,并且可以进行加减乘,如果想在当前行上加n,只需要
代码语言:perl复制=ling(".") n
先试一下,将首行的2
改成行号×10
:s/2/=line(".") * 10/
首行的2
成功被替换成了10
准备工作结束,下面直接直接给出批量更改的正则表达式
代码语言:perl复制:%s/d/=line(".")/
:%s/d$/=line(".") 1/
执行第一个指令的结果
随后执行第二个指令的结果
大功告成
最后再讲一下这两个指令的含义吧
首先是第一个指令
代码语言:perl复制:%s/d/=line(".")/
%s:所有行进行替换
d:第一个数字
=line("."):获取行号
第二个指令
代码语言:perl复制:%s/d$/=line(".") 1/
%s:所有行进行替换
d$:最后一个数字,$是位置元字符,表示行尾
=line(".") 1:获取行号并 1