菜鸟vimer成长记——第2.4章、cmd-line模式

  cmd-line模式又有3个类型:Ex 命令(ex commands)、查找模式(Search patterns)、Filter 命令(Filter commands)。本文主要重点的是Ex 命令和Search patterns。Filter 命令暂时用的场景比较少,如果有更适合的场景再补充。

目的

掌握cmd-line模式下常用操作的语法和概念

Ex 命令简介

  初时,先有ed,ed为ex之父,ex为vi之父,而vi为vim之父。可见Ex 命令的重要性。

  在Ex 命令影响范围广且距离远。对比normal模式的文本操作,它适合在本地(或者说可以通过快捷键一次到达)进行操作。

  一般操作都是范围({range})+动作。

  Vim 为几乎所有功能都提供了相应的Ex 命令。下面简要列举几个

命令用途
:[range]delete [x] 删除指定范围内的行[到寄存器x 中]
:[range]yank [x] 复制指定范围的行[到寄存器x 中]
:[line]put [x] 在指定行后粘贴寄存器x 中的内容
:[range]copy {address} 把指定范围内的行拷贝到{address} 所指定的行之下
:[range]move {address} 把指定范围内的行移动到{address} 所指定的行之下
:[range]join 连接指定范围内的行
:[range]normal {commands} 对指定范围内的每一行执行普通模式命令{commands}
:[range]global/{pattern}/[cmd] 对指定范围内匹配{pattern}的所有行,在其上执行Ex 命令{cmd}
:[range]substitute/{pattern}/{string}/[flags] 把指定范围内出现{pattern}的地方替换为{string}

  概念:

  • [range]指的是范围。下方会涉及到
  • [x] 表示的寄存器具名。即[a-zA-Z]
  • {address}表示的是地址。
  • {pattern}对应的是下面的模式
  • [cmd]表示动作
  • {string} 字符串
  • [flags] 标记
  • {commands}表示normal模式下的一系列操作

  想了解更多,参见:h ex-cmd-index。

移动光标

  在cmd-line模式下怎样更快捷的移动光标。一般是为了更正输入的错误。

  语法如下:

命令用途
Ctrl+B 移动到命令的行首
Ctrl+E 移动到命令的末尾
Ctrl+<Left> 向左移动一个单词
Ctrl+<Right> 向右移动一个单词
<Left> 向左移动一个字符
<Right> 向右移动一个字符

删除

  移动的时候,一般只使用Ctrl+B,Ctrl+E。如果此时输错了,之前在insert模式也说过了,最好的办法是删除重输。所以这里介绍一下删除的快捷键

命令用途
Ctrl+U 删除到行首
Ctrl+W 删除一个单词

 

快捷键映射

  如果还是更习惯移动光标,那可以把非主键区的按键映射成主键区的组合

  

  这个设置,大家可以根据自己的习惯来设置。我一般只设置单词移动,然后如果输错了直接删除重输。

历史记录

  查看命令的历史。

  

  查看历史记录的语法如下

命令用途
Ctrl+P 上一条命令
Ctrl+N 下一条命令

命令窗口

  命令行窗口就像是一个常规的 Vim 缓冲区,只不过它的每行内容都对应着命令历史中的一个条目。我们可以用k 及j 键在历史中向前或向后移动(像另一个独立的操作窗口,可以使用正常的所有模式命令),也可以用 Vim 的查找功能查找某一行。在按下<CR> 键时,将会把当前行的内容当成Ex 命加以执行

  开启命令窗口语法

命令用途
q/ 在normal模式打开查找命令历史的命令行窗口
q: 在normal模式打开 Ex 命令历史的命令行窗口
<Ctrl-f> 从cmd-line模式切换到命令行窗口

  例如

# 假设我们正在写一个简单的Ruby 脚本,然后发现每做出一个修改时,都会执行
# 下面两条命令:
# :write
# :!ruby %
# 在接连执行了几次这两条命令后,我们意识到可以简化工作过程,把这两条命令
# 合为一条。这样,以后就可以从历史中选择该完整命令并再次执行:
# :write | !ruby %
# 这些命令都已经在历史中了,所以我们不必从头输入整条命令。但要怎样才能把历史
# 中的两条记录合并成一条呢?

# 执行下面的步骤即可
q:
k
J(Shift+j)
<CR>

自动补全

  补全的语法

命令用途
tab 如同在shell 中一样,在命令行上也可以用<Tab> 键自动补全命令。
Shift+tab 要想反向遍历补全列表
Ctrl+D 命令会让Vim 显示可用的补全列表。这个在挺好用的。
Ctrl+A 补全所有匹配列表
ctrl+L 只有一个匹配的时候。暂时觉得用处不大
<C-r><C-w> 把当前单词插入到命令行
<C-r>{register} 可以把寄存器的内容插入到命令行

  例如

  

重复执行

  在normal模式重复上次的 Ex 命令非常简单,只需按@:。

  注意事项

  •  : 寄存器总是保存着最后执行的命令行命令。在运行过一次@: 后,后面就可以用@@ 命令来重复它。

范围

  很多 Ex 命令可以用[range] 指定要操作的范围。我们可以用行号、位置标记或是查找模式来指定范围的开始位置及结束位置。

  在定义一个[range]时,它总是代表一系列连续行,不过:global 命令也可以在一系列非连续行上执行Ex 命令

用地址指定范围

  指的就是上面的{address}。

  地址类型如下

符号地址
0 虚拟行,位于文件第一行上方。
n 文件的第n行
$ 文件的最后一行
. 光标所在的那一行
'm 包含位置标记m 的行
% 整个文件(:1,$ 的简写形式)

  注意事项:

  • 第0 行在文件中并不真实存在,但它作为一个地址,在某些特定场景下会很有用处。特别是,在把指定范围内的行复制或移动到文件开头时,可以用它做:copy{address} 及:move {address} 命令的最后一个参数。

  • 很多情况下不写[range]情况是默认当前行
  • :{start},{end}表示一个更广的范围。其中{start}和{end}都表示{address}。比如:.,$表示当前行到文件尾。

  特殊场景:

  • 如果输入一条只包含数字的Ex 命令,那么 Vim 会把这个数字解析成一个地址,并把光标移动到该数字所指定的行上。比如:3 跳到第3行。

  

用高亮选区指定范围

  当高亮选中后,按下:时,就会触发选定范围。此时在Ex 命令会显示":'<,'>"。

  '<,表示高亮选中的开头,'>表示高亮选中的结尾。

用模式指定范围

  Vim 也接受以模式作为一条 Ex 命令的地址。 :/{pattern1}/,/{pattern2}/。

  例如:

  

  这个范围看起来比较复杂,但实际上它符合范围的一般形式:{start},{end}。在本例中,{start} 地址是模式/<html>/,而{end} 地址是/<\/html>/。换句话说,这个范围由<html> 开标签所在的行开始,到对应闭标签所在的行结。  

  在此例中,用地址:2,5 也可以获得同样的结果,并且这种表示方式更简洁,不过它也更不可靠。用模式指定范围的话,我们的命令总是对整个<html></html> 范围进行操作,无论这个范围包含多少行都没问题。

  

用偏移对地址进行修正范围

  接着上面的例子,假设我们想对位于<html></html>之间的每一行都运行一条 Ex 命令,但是不想包括<html> 及</html> 标签所在的行,那么此时偏移量就有大用了。

  

  偏移的一般形式是这样的:":{addr}+n"。

  {addr} 可以是一个{address} 或是一个查找模式。可以+n,也可以减-1。"+"表示向下偏移,"-"表示向上偏移

动作

复制

  语法::[range]t{address}  t==(t|copy|co)

  例如:

  注意事项:

  • 在上表中,也可以将yyp 变化一下来复制我们想要的行,但不管怎样,这都需要一些额外的移动动作。我们得先跳到想复制的行上(6G),复制该行(yy),快速跳回原先的位置(<C-o>),然后再用粘贴命令(p)创建一个副本。由此可见,在复制距离较远的行时,:t 命令通常更加高效。

移动

  语法::[range]m[ove]{address}

  使用方法同复制非常像,就不单作解释了。

显示

  想把匹配到东西显示出来的时候可以使用这个动作。

  语法::[range]p[rint]     

连接

  把文本连接成一行的时候可以使用这个动作。

  语法::[range]j[oin]

删除

  语法::[range]d[elete][x]      

  可以把删除的内容放到寄存器x里。

  读取文本到当前缓存

  语法1:[range]r[ead]  !{cmd}  这里的{cmd}指的是外部的命令

  语法2:[range]r[ead]  {filename}  这里的{filename}指的是文件名称

  注意事项:

  • 这里的[range]如果是一个范围的话,只会取最后的位置

  把当前的缓存写到其他地方

  这里的{filename}指的是文件名称

  语法如下

命令用途
:[range]w[rite]  {filename} 把范围的文本写到指定的文件里。如果文件存在,不能直接覆盖
:[range]w[rite]!  {filename} 把范围的文本写到指定的文件里。如果文件存在,直接覆盖
:[range]w[rite]  >> 把范围的文本追加到当前文件尾。不过在窗口不会刷新出来。
:[range]w[rite]  >> {filename} 把范围的文本追加指定的文件尾。
:[range]w[rite]  !{cmd} 把范围文本当作标准输入文本调用命令

  注意事项:

  • 如果没有[range]和{filename},相当于覆盖当前文件

normal

  在一系列连续行上执行一条普通模式命令。此命令在与. 命令或宏结合使用时,我们只需花费很少的努力就能完成大量重复性任务。

  语法::[range]normal {commands}   

  注意事项

  • {commands}是普通模式的命令

  例如

#想给所有行添加注释

#使用normal
:% normal I#

#使用替换
:%s /^/#/g

global

  结合了Ex 命令与Vim 的模式匹配这两方面能力。凭借该命令,我们可以在某个指定模式的所有匹配行上运行Ex 命令。就处理重复工作的效率而言,global 命令是除点范式以及宏之外,最为强大的Vim 工具之一。

  语法::[range] global[!] /{pattern}/ [cmd]

  注意事项

  • 在缺省情况下,:global 命令的作用范围是整个文件(%),这一点与其他大多数Ex 命令(包括 :delete、:substitute 以及:normal)有所不同,这些命令的缺省范围仅为当前行(.)。

  • 反选使用:vglobal 或简写的:v。想对不匹配的内容操作的时候,可以使用这个命令。比如不匹配的全部删除::v/{pattern}/d
  • :print 是:global 命令的缺省[cmd]

查找

  语法:(/|?){pattern}

方向

  • 以/开头的,向后查找
  • 以?开头的,向前查找

重复查找

  • n     继续相当的方向查找
  • N     取相反的方向查找

关闭高亮功能的快捷键

  :noh <CR> 虽然可以禁用查找高亮功能,但我们在键盘操作上也花费了不少功夫。通过创建映射项,可以让我们加速操作。

  例如:

  nnoremap <silent> <C-l> :<C-u>nohlsearch<CR><C-l>

  <C-l> 通常用于清除并重绘显示屏(参:h CTRL-L )。而新的映射项,是在原有基础之上增加了暂时关闭查找高亮的功能。

  这个用起来很带感哈。

在执行查找前预览第一处匹配  

  每次查找的时候定位到第一处匹配的位置。按esc的时候回到原来的位置。

  开启语法::incsearch

  可以用于检查是否存在一处匹配。

  例如

  假设我们只想确认单词“carrot”是否在当前文档中出现,却不想移动光标,该怎么办呢?
  当‘incsearch’选项被启用时,我们只需简单地调出查找提示符,并尽可能多地输入组成单词“carrot”的字符,直到该单词首次映入我们的眼帘。
  一旦找到该单词,我们只需按下<Esc> ,即可马上结束查找并返回原位,从而避免打断我们的思维。

 

根据预览结果对查找域自动补全

  此法会用当前预览的匹配结果对查找域进行自动补全。单词长的时候特别好用!!力荐

  语法:<C-r><C-w> 

  例如

#文本里有一个单词like。

#查找步骤
/li <C-r><C-W> #此时会显示/like

 

将光标偏移到查找匹配的结尾

  会把光标定位到查找匹配的字符的结尾位置。如果不加的话会放在开始位置。

  语法:/{pattern}/e

  注意事项

  • 这里的"/"可以换成"?",不过两个"/",都必须换。就像这样:?{pattern}?e

查找当前选中的文本

  可以通过寄存器来实现。

  步骤如下:

  1. 用visual模式选中要查找的文本
  2. 然后使用y复制到寄存器""里。
  3. 最后输入/<C-r>"<CR>

替换

  语法::[range]s[ubstitute]/{pattern}/{string}/[flags]

重用上次的查找模式

  执行substitute 命令通常包括两个步骤:一是撰写查找模式,二是设计合适的替换字符串。因此,一分为二的技术让我们消除了这两项任务的耦合性,这才是关键所在。

  将substitute 命令的查找域留空,意味着Vim 将会重用上次的查找模式。我们可以利用这一特点精简工作过程。

  例如

# 看看这个庞大的substitute 命令吧
:%s/\v'(([^']|'\w)+)'/“\1”/g

#它等价于以下两条单独的命令:
/\v'(([^']|'\w)+)'
:%s//“\1”/g

# 在我们撰写复杂的正则表达式过程中,通常需要尝试多次才能达到正确的匹配效果。
# 如果打算通过执行substitute 命令的方式来验证模式的话,每次执行命令都会改变文档的内容,这样做简直太麻烦了。
# 与之形成鲜明对比的是,当执行查找命令时,文档不会被修改。因此,即使我们犯的错误再多也无所谓


# 请注意,这个一般用于复杂的匹配。。如果很简单的匹配,就不要这么弄了。这个简单因人对正则的熟练度。
# 比如,5,10行加注释,这个简单一句话就可以解决了。
:5,10s/^/#/g

 

用寄存器的内容替换

  输入入<C-r>{register},我们可以将寄存器的内容插入到命令行 

标记

  {flags}。不同的标记影响替换的行为。

  具体功能如下

字符行为
& 保持上一个替换的标记
c 需要进行确认
e 如果有错误,不提示错误信息。暂时没有找到比较不错的场景
g 作用于全部匹配的内容
i 忽略大小写字母敏感
I 大小写字母敏感
n 统计计数,不执行替换内容。
p 打印出最后一次匹配的内容。如果有设置显示行号,则显示,没有设置则不显示行号
# 打印出最后一次匹配的内容,并且显示行号。
l 打印出最后一次匹配的内容,但是打印的内容和:list一致。按我理解是多打印出逃逸字符

   技巧:

  统计当前模式的匹配个数   

/{pattern}                #这个就是我们要统计的模式
:%s///gn           #抑止执行

模式

  即{pattern}。

  让我们先把目光集中在驱动它们运行的核心上,即Vim 的搜索引擎。你是否曾经想过Vim 的正则表达式是如何工作的?或者怎样关掉它们?

  Vim 的正则表达式引擎可能与你惯用的其他引擎有所不同。我们将会看到,最易混淆的差异可被very magic 模式开关轻松化解。

大小写敏感

  一般默认的是大小写敏感的。如果需要强制,在字符串后面加上标志即可。

  • \c(小写c),表示大小写不敏感
  • \C(大写c),表示大小写敏感

  例如

#查找单词"word",大小写不敏感
/word\c
#查找单词"word",大小写敏感
/word\C

#把单词“word"替换为"me",大小写不敏感
:%s/word\c/me/g
#把单词“word"替换为"me",大小写敏感
:%s/word\C/me/g

 

vim的正则

  与Perl 相比,Vim 正则表达式的语法风格更接近POSIX。

  但是,通过使用very magic 模式开关,就可以让Vim 采用我们更为熟悉的正则表达式语法了。

  缺省使用的是magic模式。通过\v(小写v)可以使用very magic模式。通过\V(大写v)可以very nomagic模式。

例如

#假设我们要构造一个正则表达式,用于匹配以下CSS 片段中的每一组颜色代码:
body { color: #3c3c3c; }
a { color: #0000EE; }
strong { color: #000; }

#用magic 搜索模式查找十六进制颜色代码
/#\([0-9a-fA-F]\{6}\|[0-9a-fA-F]\{3}\)

#用very magic 搜索模式查找十六进制颜色代码
/\v#([0-9a-fA-F]{6}|[0-9a-fA-F]{3})

#用十六进制字符类进一步优化模式
/\v#(\x{6}|\x{3})

 

  注意事项

  • 在此例中,magic模式我们用到了3 类括号。方括号缺省具有特殊含义,因此不用转义。圆括号会按原义匹配字符(及),因此需要转义,使其具有特殊含义。花括号也一样需要转义,不过,我们只需为开括号转义,而与之对应的闭括号则不用,因为Vim 会推测我们的意图。圆括号的情况有所不同,无论开闭括号都必须转义。

magic模式

  模式会自动为某些额外的符号赋予特殊含义,例如:. 、* 以及方括号。magic模式的设计初衷,是想能更容易地构造简单的正则表达式,但它却没能为诸如 +、?、圆括号以及花括号等符号赋予特殊含义,这些符号还必须经过转义才具有特殊含义。

nomagic模式

  暂时还不是很清楚???

very magic模式  

  开关正好弥补了这一点,除了 _、数字以及字母外,它为所有符号都赋予了特殊含义。这样一来,既好记又恰好与Perl 正则表达式的规则保持一致。

very nomagic模式

  在正则表达式中使用的特殊字符,在按模式查找时用起来很顺手,但如果我们想按原义查找文本时,它们就变成了阻碍。使用very nomagic 原义开关,可以消除附加在 .、* 以及? 等大多数字符上的特殊含义。

  例如:

#文本内容
The N key searches backward...
...the \v pattern switch (a.k.a. very magic search)...

#现在假设我们想通过查找“a.k.a.”(此缩写表示also known as)的方式将光标移到
该处。针对这种情况,第一反应就是执行以下这条查找命令:
/a.k.a.

#实质上我们需要这么才能得到效果
/a\.k\.a\.<CR>

#或者,我们可以使用原义开关\V,激活very nomagic 搜索模式:
/ \Va.k.a.

 

  注意事项

  • 作为通用法则,如果你想按正则表达式查找,就用模式开关\v,而如果你想按原义查找文本,就用原义开关\V。这个要特别注意

特殊字符

  在几种模式的编写时,特殊字符比较容易混淆。

  几种模式的特殊字符对应列表如下:

magicnomagicvery magicvery nomagic匹配内容
$ $ $ $ 行尾
. . \. \. 任何字符
* * \* \* 任何个数
() \(\) \(\) \(\)
| \| \| \| 分隔符
\a \a \a \a 字母表的字符
\\ \\ \\ \\ 反斜杠
\. \. . . 字符点
\{ { { { 字符{
a a a a 字符a

  

匹配和高亮 

  当我们谈论一个模式的时候,指的是在查找域输入的正则表达式(或者按原义匹配的文本);而匹配,是指在文档中被高亮显示的文本(假设已经启用'hlsearch'选项)。匹配和高亮是两件事。

  有这么个东东,但是暂时没觉得有什么用处,试了一下暂时没体会到运用场景。

  语法:

  • /zs  高亮开始
  • /ze  高亮结果

运行Shell 命令

  我们不用离开 Vim 就能方便地调用外部程序。更棒的是,我们还可以把缓冲区的内容作为标准输入发送给一个外部命令,或是把外部命令的标准输出导入到缓冲区里。

  在 Vim 中操作时,我们能很方便地调用shell 命令。下表选取了最有用的一些调用外部命令的方式

命令用途
:shell 启动一个shell (输入exit 返回Vim)
:!{cmd} 在shell 中执行{cmd}
:read !{cmd} 在shell 中执行{cmd} ,并把其标准输出插入到光标下方
:[range]write !{cmd} 在shell 中执行{cmd} ,以[range] 作为其标准输入。这个暂时还没试明白?
:[range]!{filter} 使用外部程序{filter} 过滤指定的[range]

  运行shell命令一共有如下几种类型

一次性外部

  适用于执行一次性命令。

  语法::!{cmd} 

交互

  适用于想在 shell 中执行几条命令。

  语法::shell

  注意事项

  • 想要退出的时候,输入exit,即可返回vim

使用外部命令过滤缓冲区内容

  语法::[range]!{filter}

  例如

#把当前文本排序一下

:% !sort

 

把命令的标准输出重定向到缓冲区  

  语法::[range]read !{cmd}

  例如

#文件开头插入当前时间

:0r !date

 

  注意事项

  • 如果没有[range],则插入到当前光标位置
  • 如果有[range],以range后面的匹配的范围行为准
posted @ 2015-08-10 18:33  庄君祥  阅读(967)  评论(2编辑  收藏  举报