想法子记忆Vi/Vim常用操作及指令

虚幻大学 xuhss 204℃ 0评论

? 优质资源分享 ?

学习路线指引(点击解锁) 知识定位 人群定位
? Python实战微信订餐小程序 ? 进阶级 本课程是python flask+微信小程序的完美结合,从项目搭建到腾讯云部署上线,打造一个全栈订餐系统。
?Python量化交易实战? 入门级 手把手带你打造一个易扩展、更安全、效率更高的量化交易系统

本笔记有特殊目录,点击开启:

在Linux系统中编辑文本总是离不开一位老帮手——Vi。而因为其诞生的年代有些久远,有些操作在现在看来可能有点“反直觉”。

于是我决定写这样一篇小笔记,记录一下我记忆Vi的这些这些 常用 操作和指令的方法(主要靠的是英语和图示了)。

当然,正如“好记性不如烂笔头”所言,多实践才是熟练掌握技能的王道。

1a7e8a084d1e1bea3fb6f292434c3db5 - 想法子记忆Vi/Vim常用操作及指令

关于Vi/Vim名词本身

  • Vi代表的可能是Visual Instrument(可视工具)Visual Interface(可视界面)亦或就是Visual(可视化的)单词本身。
  • Vim倒是很明显,代表的是Vi IMproved(Vi改进版)

编辑器模式

Vi而言:

  1. 命令模式 (Command mode) ,刚进入Vi编辑器时的默认模式。在此模式下能输入Vi的合法指令(通常是一些字符)来进行操作。
  2. 插入(输入)模式 (Insert mode) ,顾名思义,是直截了当地编辑文本的模式。
  3. 末行(底线)模式 (Last-line mode) ,也是接受指令的一种模式,更偏向于文件的管理。
    之所以叫 “末行” 是因为指令的输入在窗口最下方:

e8a1079820229b97e9bf2e9988b16e78 - 想法子记忆Vi/Vim常用操作及指令

VimVi的增强版,因此Vim完全可以兼容Vi的操作,不过它的模式名可能略有差别:

  1. 正常模式 (Normal mode) ,是启动Vim后的默认模式。
  2. 插入模式 (Insert mode) ,这个和Vi的一致。
  3. 命令(行)模式 (Command-line mode),类似Vi的末行模式。
  4. 可视模式 (Visual mode) ,和普通模式类似,不同的是可以高光选择文字的一部分进行命令操作

包括上述几种模式,Vim总共有12种模式(这个说法来自Wikipedia)。但最常用的还是上面这四种,一般情况下够用了。

0b312f31adece3648df7a702fabb1a2c - 想法子记忆Vi/Vim常用操作及指令

模式切换

展开阅读


Vi的三种模式而言,要在其间互相切换,需要借默认模式——命令模式作为一个中转(对于Vim来说就是正常模式)。

  • 命令模式

作为进入Vi的默认模式,我们可以把这儿当作一个小家,
无论在哪个模式下Esc(可以记成Escape [v.]逃离。逃回家了)都可以回到命令模式。(家是永远的港湾啊)

dc10a471db842ac1356d166ba8248958 - 想法子记忆Vi/Vim常用操作及指令

再提一嘴,在Vim里这个模式称为正常模式

  • 插入模式

通过以下按键指令能进入插入模式。这些按键通常是大小写成对的,小写对应的指令略显 “温和”大写对应的就比较 “走极端” 了。

d3db2848eb9cf5bbe8d601de4bf0ac7c - 想法子记忆Vi/Vim常用操作及指令

1. i / I 插入 (记作Insert [v]插入;嵌入;(在文章中)添加 )

小写的i,就是在**光标所在位置**进入插入模式:

[![lowerInsert-2022-04-07](https://img-blog.csdnimg.cn/img_convert/1a4854036086a739a20dd5a7b0d01db5.gif)](https://blog.csdn.net/biggbang)

大写的I则有点“极端”,会**跳转到当前光标所在行的开头**进入插入模式:

[![upperInsert-2022-04-07](https://img-blog.csdnimg.cn/img_convert/036921393fff89ba51f42ed13da8ca7e.gif)](https://blog.csdn.net/biggbang)
2. a / A 附加 (记作Append [v]增补,追加)

小写的a,就是在**光标所在位置之后一位**进入插入模式,即所谓的“附加”:

[[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0P8m1qOo-1650734010030)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/lowerAppend-2022-04-07.gif)]](https://blog.csdn.net/biggbang)

显而易见,大写的A就会**跳转到当前光标所在行的末尾**进入插入模式(在尾部附加):

[![upperAppend-2022-04-07](https://img-blog.csdnimg.cn/img_convert/3c2014953c5792db8f3c42403a5f9a17.gif)](https://blog.csdn.net/biggbang)
3. o / O 新增一行 (暂且记作Open new line)

都是新增行,差别无非是往下新增还是往上新增了。

小写的o,就是在光标所在行的**之后新建一行**,然后进入插入模式:

[![newLineAfter-2022-04-07](https://img-blog.csdnimg.cn/img_convert/0eb03d3d809f4cf75e42f3042e39a871.gif)](https://blog.csdn.net/biggbang)

大写的O呢,就比较蛮横,会把光标所在行及其以下的行整体往下顶,在原位置新建一行(相对来说就是**在光标所在行上面新建一行**),然后进入插入模式:

[![newLineBefore-2022-04-07](https://img-blog.csdnimg.cn/img_convert/aba16f27a48200b089277384aeea9365.gif)](https://blog.csdn.net/biggbang)
4. s / S 剪切(取代) (可以直接记删的拼音Shan,也可以记成Substitute [v]取代)

小写的s,会**剪切光标所在位置右边的字符**,然后进入插入模式:

[![subWord-2022-04-07](https://img-blog.csdnimg.cn/img_convert/16f1a34640d94d24c421f72d18aadd23.gif)](https://blog.csdn.net/biggbang)

大写的S就很暴躁了,会**剪切光标所在行**,然后进入插入模式:

[[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-zIYkOY0M-1650734010036)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/subLine-2022-04-07.gif)]](https://blog.csdn.net/biggbang)

> 
> 记Substitute的话,可以理解为这个操作是剪切原有字符/行后进行插入,**取代**原有的字符/行。
> 
> 
> 

? Scc 指令的功能是相同的!这个在下面的[**“剪切一行”**](https://blog.csdn.net/biggbang)这一节中也提到了。上面这些按键指令都是在命令模式/正常模式下输入的,通过**在指令前附加数字**,能重复这个指令的操作:

+ 3s -> 删除光标后3个字符

[[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MZgDtXra-1650734010036)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/repeatSubstitute-2022-04-07.gif)]](https://blog.csdn.net/biggbang)
+ 4a -> 在光标下一位附加内容,**操作结束后**(Esc)把这个内容重复4次

[![appendFourTimes-2022-04-07](https://img-blog.csdnimg.cn/img_convert/07872d98e543b14a0491fc2c2af158f7.gif)](https://blog.csdn.net/biggbang)
  • 替换模式

这个模式和插入模式一样是用于编辑文本的,不同的是替换模式下输入所至之处会覆盖一切。

记作 Replace [v]取代;代替。

输入大写R进入替换模式:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6xWH036t-1650734010038)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/replaceMode-2022-04-07.gif)]

有了大写R,那么小写r呢?小写r的功能是将光标后面的字符替换为指定字符。比如我们先输入r,再输入a:

db0f3f1c229cfaa610a0ce1d74268886 - 想法子记忆Vi/Vim常用操作及指令

当然,这些操作一样是可以重复进行的:

35320f4398d44284158b08c1800b99a2 - 想法子记忆Vi/Vim常用操作及指令

(指令分别是3rp3R->hello->Esc)

  • 末行模式(Vim中的命令行模式)

输入 Shift+: 进入末行模式(事实上我们一般都是这样打出冒号来的)

末行模式在Vim中被称为命令(行)模式

  • 可视模式

可视模式是Vim中新引入的模式,支持在高光选择下进行指令操作。

可视模式即 Visual Mode,进入可视模式的按键则是v / V

其中 v 键能普通地进入可视模式:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Efg3LT6d-1650734010041)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/visualNormal-2022-04-07.gif)]

V 在进入可视模式后会保持高光选择到光标所在行可视行模式):

03fec753ecc5babb4af9d19869d88639 - 想法子记忆Vi/Vim常用操作及指令

还有一种便是使用 Ctrl+V/v 进入可视模式了,这种模式叫可视块模式,如字面所言,是选择“一块文字”:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-na2YDOVl-1650734010043)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/visuallySelectBlock-2022-04-23.gif)]

命令模式/正常模式

这是进入Vi/Vim后所处的默认模式,和文本处理是靠的非常近的,我决定先记一下这部分。

  1. 模式切换
  2. 基本光标移动(上下左右)

在交互式字符界面下没有能动的鼠标,只有闪烁的光标来标注位置。要编辑文本,真得先掌握移动光标的指令。

PS: 下方涉及的光标上下移动都是针对换行符\n分隔的行的,光标会从一行移动到上方或下方的行:

展开查看示例


daec1cf759e8e2bd06e30f1febac7587 - 想法子记忆Vi/Vim常用操作及指令

? jk将光标从一行的指定位置移动到另一行的指定位置

a30f8829a5636e14a8418b510e5b4652 - 想法子记忆Vi/Vim常用操作及指令

? Enter-,将光标移动到下一行或上一行的行首

* **“经典”款**

h 向左移动光标
l 向右移动光标
k 向上移动光标
j 向下移动光标

之所以这样设计,因为在早期**终端机**上,键盘是这个样的:

[![hjklOnTerminal-2022-04-08](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/hjklOnTerminal-2022-04-08.jpg)](https://blog.csdn.net/biggbang)

**记忆方法**:现代键盘上键在中间,而键在两侧,而HJKL也正是两侧的KL代表横向移动,中间的JK代表纵向移动(HJKL四个键连在一起的设计一直保留到了今天)。

[![hjkl-remarked-2022-04-08](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/hjkl-remarked-2022-04-08.gif)](https://blog.csdn.net/biggbang)
* **现代款**

 向左移动光标
 向右移动光标
 向上移动光标
 向下移动光标
* **“隐藏”款**

Backspace(退格键) 向左移动光标
Space(空格键) 向右移动光标

> 
> **?记忆方法**:在插入模式下,BackspaceSpace是分别用于删去字符和增加空格的,与之对应的光标分别是**往左移动**和**往右移动**的。
> 
> 
> 

---

Enter 把光标移动到**下一行行首**
+ 把光标移动到**下一行行首**
- 把光标移动到**上一行行首**

> 
> **?记忆方法**:插入模式下 Enter 起到了换行作用,换行后光标会处于行首。
> - 就记成“减去一行”(老实说我也不太知道这个怎么记了,JK其实也够用了),
> +- 记在一起就行。
> 
> 
> 

---**PS:** 光标移动是在命令模式/正常模式下的指令,当然是支持重复的啦!

在上面所有指令的前面加上重复的次数(数字)即可,比如我要往下翻233行: 233j233↓

  1. 行内光标移动

这里记录一下在一行文本中光标的移动指令:

* **跳到行首**

0 回到行首,列数=1的地方
Home 同上?
^ 回到行首,**第一个不为空白符**的字符处(对于加了缩进的行可谓是非常有用)

> 
> **?记忆方法**:0记为“归零”;
> ^的话,**正则表达式**里是用于匹配字符串首部的,可以一起记,另外其还形如一个顶部,可以记成一行中的“顶头”部分。
> 
> 
>
* **跳到行尾**

$ 回到行尾**最后一个字符**的地方
End 同上?
g_ 回到行尾**最后一个不为空白符**的字符处

> 
> **?记忆方法**:$在正则表达式里是用于匹配字符串尾部的,可以和正则表达式一起记;
> g_可以记成go __形如一个输入光标,这样记成跳转到行尾最后一个非空白字符。
> 
> 
>
  1. 屏幕可见行中移动光标

上面所说的“行”取决于换行符\n的位置,但这里的屏幕可见行是取决于窗口大小的:

cf3cfbe2eebdaab9470bb3bef80c3a57 - 想法子记忆Vi/Vim常用操作及指令

(图中标注的即为一个屏幕可见行

针对这样的一个行,也是有指令可用于行内移动的:

* g^ 跳转到行首第一个不是空白符的字符处

> 
> **?记忆方法**:g^可以记成go ^,去往当前的顶头部分。
> 
> 
>
* g0 跳转到行首字符处
* gm 跳转到**行中间**的字符处

> 
> **?记忆方法**:gm可以记成go midst,去往当前的中间部分。
> 
> 
>
* g$ 跳转到行尾字符处完全是可以结合前面的0^$一起记的。当然,g + Homeg + End也是可以的。

再展开想想,既然g开头的指令在这里看的是屏幕可见行,我是不是可以用gj(或g↓)指令从一个屏幕中的行移动到另一个行呢?

08da1468b1aeb4190eddfdc6edb18c69 - 想法子记忆Vi/Vim常用操作及指令

?这个主要是用j)指令进行的光标移动

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fifTYxWD-1650734010047)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/gjOperation-2022-04-10.gif)]

?而这个主要是用gjg↓)指令进行的光标移动。

利用g+移动指令,可以实现在屏幕中的行之间移动。当然,3gj这种移动多行的指令也是支持的哦~


针对屏幕可见行,还有几个指令:

* 大写 H 用于将光标移动到**屏幕可见**的**最上面一行**(这个判定有点怪,可能不太准确!)

> 
> **?记忆方法**:H可以记成Head,去往可见区的头部。
> 
> 
>
* 大写 M 用于将光标移动到**屏幕可见**的**最中间行**(同样判定很怪)

> 
> **?记忆方法**:M可以记成Middle,去往可见区的中间部分。
> 
> 
>
* 大写 L 用于将光标移动到**屏幕可见**的**最下面一行**

> 
> **?记忆方法**:L可以记成Last,去往可见区的最后一行。
> 
> 
>
  1. 文件内光标移动

除了上述几种移动外,还有几个常用指令能按词、按句、按段等移动光标:

* **按词移动**

w 跳转到**下一个词**的开头部分(**考虑标点符号**),演示如下?

[![wordNext-2022-04-10](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/wordNext-2022-04-10.gif)](https://blog.csdn.net/biggbang)

W 跳转到**下一个词**的开头部分(只考虑**空格分隔**) ,演示如下?

[![wordNext-CAPS-2022-04-10](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/wordNext-CAPS-2022-04-10.gif)](https://blog.csdn.net/biggbang)

b 跳转到**上一个词**的开头部分(**考虑标点符号**)。

B 跳转到**上一个词**的开头部分(只考虑**空格分隔**)。

这里的bBwW的操作就是**反着来的**。

> 
> **?记忆方法**:w可以记成word next,下一个单词;
> b则记成behind word,后一个单词。
> WB则比较“激进”,不会考虑标点符号。
> 
> 
> 

---

e 跳转到当前**光标所在词**的**词尾**,如果光标在词尾,就跳转到**下一个词的词尾** (**考虑标点符号**)。

E 同上?,但是**只考虑空格分隔**。

> 
> **?记忆方法**:e可以记成end of word,词尾。
> 
> 
>
* **按句移动**

( 移动到**光标所在句子的开头部分**,如果已经在开头,就移动到**上一个句子**的开头部分

) 移动到**下一个句子**的开头部分

> 
> **?记忆方法**:写代码的时候会用半角括号()把表达式括起来,可以理解为“一句”。
> 
> 
>
* **按段落移动**

{ 移动到**光标所在段落的开头部分**,如果已经在开头,就移动到**上一个段落**的开头部分

} 移动到**下一个段落**的开头部分

> 
> **?记忆方法**:写代码的时候会用半角大括号{}把代码块括起来,也就成为了一“段”代码,可以理解为一个段落。
> 
> 
>值得注意的是,上面这些按词、句、段落移动光标的操作,主要是**针对英文句式**的。如果操作对象是中文的话,可能并不能如愿。

(最明显的就是按词移动 W,中文里一般是不会用空格分隔每个词的,所以会直接跳转到一段的末尾!)


* **在整个文件中跳转光标**

⚠️ 下面的指令全部都会使光标**跳转到行首第一个不为空白符的字符**处。

gg 跳转到文件**最开头的一行**,相当于1GG 跳转到文件**最末尾的一行**。

[row]G 跳转到文件**第[row]行**(row代表数字)。

[![GJump-2022-04-10](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/GJump-2022-04-10.gif)](https://blog.csdn.net/biggbang)

? 分别演示了 ggG2G 的操作。

> 
> 记忆方法:上面三个指令本质都和G有关,G可以直接记忆为Goto 转到,跳转。
> 
> 
> 

[![nGJump-2022-04-10](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/nGJump-2022-04-10.webp)](https://blog.csdn.net/biggbang)
  1. 自定义跳转标记

在这一坨文本中进行移动时,我经常要使用指令反复横跳。有一个地方我经常跳转过去,如果一步步来实在很麻烦!此时就可以用自定义跳转标记来解决。

* **设置标记**

指令非常简单:

ma

这样就设置了一个名为a的标记!

经过测试发现 m 后面可以接 a-zA-Z0-9 之中的任意字符!

> 
> 也就是说m1, mA, mO, mz这些都是合法的指令!
> 
> 
> 

> 
> ?记忆方法:m 可以直接记忆为 Mark [v]标示,标记。
> 
> 
>
* **跳转到标记处**

`a

这样就能跳转到a标记处~

标记名是**大小写敏感的**。

> 
> ? 可以和下面的y, c, d指令联合使用。
> 
> 
>
* **跳转到标记所在行的行首**

'a

使用单引号 ' 加标记名能跳转到**该标记所在行**的行首**第一个不为空白符的**字符处!

[![jump2markedLine-2022-04-12](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/jump2markedLine-2022-04-12.gif)](https://blog.csdn.net/biggbang)

> 
> 这样是不是就相当于标记了一个行的行首呢?
> 
> 
>
* **删除标记**

ma可以设置名为a的标记,删除的话其实在前面加个del即可:

delm adelmarks aDelete Marks )

同时支持**删除多个标记**。比如我要删除a,b,c三个标记,可以写成:

delm abcdelmarks a b c

也就是说删除标记的时候**标记名之间有没有间隔是无所谓的**。

? 在进行了缩进操作后,标记的位置仍然在文本的相应位置,无需担心。

末行模式/命令行模式下可以查看标记信息:marks

  1. 翻页

    • 后空翻

    Ctrl + b 往回翻一整页
    PageUp 往回翻一整页
    Ctrl + u 往回翻半页

    ?记忆方法b 代表 Backward,也就是往回;
    u则代表Up,也就是往上(在界面上往回翻就如同往上翻);
    PageUp不多赘述,在现在的大多系统中都是支持的。

    • 前空翻

    Ctrl + f 往前翻一整页
    PageDown 往前翻一整页
    Ctrl + d 往前翻半页

    ?记忆方法f 代表 Forward,也就是往前;
    d则代表Down,也就是往下(在界面上往前翻就如同往下翻)。

    Ctrl键代表Control,在这里可以理解为控制页面滚动。

PS: 再记忆一下,命令模式/正常模式下的指令是可以重复的。在这里如果我要向前翻3整页:3PageDown3Ctrl+f

  1. 复制

复制是文本的经典便捷操作之一。

* **复制一行**

yy 复制光标所在行(到寄存器中)
Y 同上

> 
> **?记忆方法**:yy 中的 y代表yank [v]猛拉;猛拽,复制其实就是把对应的文本给“拽走”了。
> 
> 
>
* **多行复制**

2yy 复制**光标所在行**以及**下一行**,共22Y 同上
10yy 复制**光标所在行**以及**下面9行**,共10行

> 
> 本质上其实就是命令模式/正常模式下的指令重复(在指令前面加数字前缀)。
> 
> 
>
* **打一套复制组合拳!(选区复制)**

y其实可以和很多**光标移动指令**结合使用!利用好这套“组合拳”,效率倍增有木有!

[![yankBoxing-2022-04-10](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/yankBoxing-2022-04-10.png)](https://blog.csdn.net/biggbang)

比如 ye 就能复制光标所在位置到[**单词尾部**](https://blog.csdn.net/biggbang)的部分,

而 y^ 能复制光标所在位置[**到行首**](https://blog.csdn.net/biggbang)第一个非空白符的单词部分,

再者 y↓yj 能实现同2yy一样的功能:**复制光标所在行**以及[**下一行**](https://blog.csdn.net/biggbang),

甚至 y( 都是可以的,复制**光标所在位置**到[**光标所在句开头**](https://blog.csdn.net/biggbang)。

> 
> 也就是说 y{, y}, yk ,yl ,y$ ,yg^, ya` 等等其实都是可以的!
> 另外还可以复制多行,比如3ygj。
> 
> 
> 

---

? **总结一下**就是:

y 可以和很多**光标移动指令**结合使用达成**部分复制**的效果,

通常以**光标所在位置**为起点,直到**光标移动指令执行后的光标位置**为止作为**选区**进行复制。

[![yankGoTest-2022-04-11](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/yankGoTest-2022-04-11.gif)](https://blog.csdn.net/biggbang)

?示例ygj的效果。光标所在位置是字前面,执行gj后光标位于字前面,所以复制选区就是:

语人曰:“此当有大手笔事。”俄而帝崩,哀册说议皆淘所
  1. 剪切

有了复制,就不得不提到复制并移除——剪切了。通过这些指令移除的文本会被存进寄存器,可用于粘贴

* **剪切一行**

dd 剪切**光标所在行**
cc 剪切**光标所在行**,并且**进入插入模式**
S 同上?,这个其实在[**模式切换**](https://blog.csdn.net/biggbang)这一节里面写了

> 
> **?记忆方法**:dd 中的 d记为delete [v]删除;
> cc中的c记为change[v]改变,亦可记成Chi 吃;
> S可以记成Shift [v]转移,剪切就像是在转移文本的位置,亦可记成Shan 删。
> 
> 
>
* **剪切多行**

这一部分和复制多行真的相差无几,可以说会用[**多行复制**](https://blog.csdn.net/biggbang)了,肯定是会剪切多行的。

2dd 剪切**光标所在行**和**下面一行**,共26dd 剪切**光标所在行**及**下面5行**,共63cc 剪切光标所在行及下面2行,并进入**插入模式**
3S 同上?
* **剪切一个字符**

x 剪切**光标右边的一个字符** (相当于dld→s 剪切**光标右边的一个字符**,并且**进入插入模式**

> 
> **?记忆方法**:x 中的 x记为画个叉叉删掉字符;
> s则可以记为shan 删。
> 
> 
>
* **行内剪切**

和复制不同的是,大写 DC 的作用**并不等同于** ddccD 剪切从**光标所在位置**到**光标所在行的行末**部分
C 完成上述操作?,并且进入**插入模式**

> 
> C相对D仍然是多出了个**进入插入模式**的功能
> 
> 
>
* **组合剪切(选区剪切)**

复制y能打组合拳,我堂堂剪切为什么不行!

确实,剪切 dc 都是可以和**光标操作指令**进行结合的,
他俩不同的地方在于 d 只是普通的剪切,c 还附赠了[**切换到插入模式**](https://blog.csdn.net/biggbang)服务!

[![arrogantCandD-2022-04-11](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/arrogantCandD-2022-04-11.png)](https://blog.csdn.net/biggbang)

djd↓ 的效果就等同于2dd
cjc↓ 的效果就等同于2cc,会进入插入模式
5x 能剪切光标右边的5个字符
5s 能剪切光标右边5个字符且进入插入模式
3cgj 能剪切从**当前光标**到3个**屏幕可见行**后的相应位置部分
dw 剪切从**当前光标位置**到**下一个单词前**的部分
cw 在进行上面的操作?后,会进入**插入模式**
d^ 剪切从**当前光标位置**到**本行头部一个不是空白符的字符**的部分
c^ 在进行上面的操作?后,会进入**插入模式**
da 剪切从**当前光标**位置到a`[**标记位置**](https://blog.csdn.net/biggbang)的部分

[![3dgj-2022-04-11](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/3dgj-2022-04-11.gif)](https://blog.csdn.net/biggbang)

?演示了一下 3dgj 。从光标所在位置  **开始**,到执行了3gj之后的光标位置**为止**作为选区,进行剪切。也就是这一部分:

```
 语人曰:“此当有大手
笔事。”俄而帝崩,哀册说议皆淘所草。释义用来赞誉写作才
能极高,并用来称颂着名的作家和作品、故事东晋的文士王玖
从小才思敏捷,胆量很大,散文和诗

```

---

? **总结一下**就是:

d / c 可以和很多**光标移动指令**结合使用达成**部分剪切**的效果,

通常以**光标所在位置**为起点,直到**光标移动指令执行后的光标位置**为止作为**选区**进行剪切。

> 
> ? 应该和**复制**结合记忆!
> 
> 
>
  1. 粘贴

Copy和Cut都记了,接下来咱要想办法粘贴(Paste)内容咯~

粘贴的指令其实非常简单!( ´・ω・)ノ

* p (向下一行/向右)粘贴文本一次(记为 Paste [v]粘贴)
* 大写 P (向上一行/向左)粘贴文本一次(记为 Upper(大写,同时也有“上边”的意思) Paste

寄存器中的内容是一行或多行文本(开头有换行符\n)时:

* p会把文本粘贴到当前**光标所在行**的**下一行**,
* 大写 P 会把文本粘贴到当前**光标所在行**的**上一行**!
* ?举例:先利用yy复制**一行**文本,然后分别用 pP 进行粘贴

[![pasteAndUpperPaste-2022-04-11](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/pasteAndUpperPaste-2022-04-11.gif)](https://blog.csdn.net/biggbang)

> 
> 很明显能看到使用p时,该行文本会被粘贴到line2这一行的下方;
> 而使用P时,该行文本会被粘贴到line2这一行的上方。
> 
> 
>

寄存器中的内容是不足一行的文本开头没有换行符\n)时:

* p会把文本粘贴到当前**光标所指字符**的**右边**,
* 大写 P 会把文本粘贴到当前**光标所指字符**的**左边**!
* ?举例:先执行ygj复制**开头没有换行符**的一段文本,然后分别用 pP 进行粘贴

[![ygjPasteAndUpperPaste-2022-04-11](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/ygjPasteAndUpperPaste-2022-04-11.gif)](https://blog.csdn.net/biggbang)

> 
> 粘贴的时候**光标指向的字符**是n。使用p粘贴时,内容粘贴在n的右方;而使用P粘贴时,内容粘贴在n的左方。
> 
> 
>

? 命令模式/正常模式下指令可加数字前缀重复执行,这里也是一样的。比如 4p 就是把寄存器的文本在光标右边/下一行粘贴四次

  1. Vim上下文编辑

Vim在复制y,剪切cd这些操作的基础上复用了ia指令用于对上下文进行操作:

?记忆方法:i 记作 inner 内部的,在下面使用这个指令,操作不会考虑周围的空格或符号
a 记作 around 周围,这个则会考虑周围的空格或符号。具体视情况而定。

* **针对单词**(w / W)

> 
> ⭐ 记作Word
> 
> 
> 

diw 剪切**光标所在的单词**(不考虑标点符号)

diW 剪切**光标所在的单词**(考虑周围的标点符号,也就是说剪切的时候会把周围的标点符号一并算进去)

> 
> i (inner) 在这里的意思是操作的部分**不包括单词周围的空格**
> 
> 
> 

dawdaW 也是剪切**光标所在的单词**(wW区别不再多赘述)

> 
> a (around) 在这里的意思是操作的部分**包括单词周围的空格**
> 
> 
> 

※ 下面演示一下diwdaw的区别:

[![diwAnddaw-2022-04-11](https://img-blog.csdnimg.cn/img_convert/9ef016feeb71bbf6a58c744f48d33db1.gif)](https://blog.csdn.net/biggbang)

很明显能看到,使用diw时,剪切后遗留的文本是 of ,(单词左边的空格保留了);
使用daw时,剪切后遗留的文本是 of,(单词左边的空格被一同剪切了) 。

> 
> ? 像 y, c, d 一类指令都是可以这样用的,比如yaw, yiW, ciw, caW 等等...
> 
> 
> 

---
* **针对句子**(s)

> 
> ⭐ 记作Sentence。注意,这里**没有大写** S 的用法!
> 
> 
> 

其上下文操作和上面单词的是一样的!比如:

yis 复制**光标所在的句子**(i不考虑周围的空格)
yas 复制**光标所在的句子**(a考虑周围的空格)
cas 剪切**光标所在的句子**,并进入插入模式
dis 剪切**光标所在的句子**
...and so on

> 
> ? 注意,这里只有小写s的用法。
> 
> 
> 

---
* **针对段落**(p)

> 
> ⭐ 记作Paragraph。注意,这里**没有大写** P 的用法!
> 
> 
> 

这里的用法也和上面是一样的:

cap 剪切**光标所在的段落**,并进入插入模式(a考虑周围的空格)
dip 剪切**光标所在的段落**(i不考虑周围的空格)
yip 复制**光标所在的段落**
...and so on

> 
> ? 这里同样只有小写p的用法。
> 
> 
> 

---
* **针对单/双引号**( ' / " )

很明显,这一部分操作是用来处理单/双引号**引起来的文本**的,和上面的wWsp不同的是这里的 i / a 的含义:

> 
> i (inner) 在这里的意思是操作的部分**不包括**两侧的引号
> a (around) 在这里的意思是操作的部分**包括**两侧的引号
> 
> 
> 

di" 剪切光标所在的**双引号**引起来的**同行**文本(i代表不包括两侧的引号) ?

[![diQuoted-2022-04-12](https://img-blog.csdnimg.cn/img_convert/42f28cc0e3a05587dccc30096125628c.gif)](https://blog.csdn.net/biggbang)

da' 剪切光标所在的**单引号**引起来的**同行**文本(a代表包括两侧的引号) ?

[![daQuoted-2022-04-12](https://img-blog.csdnimg.cn/img_convert/439e258a940951f82e501912a7ed537b.gif)](https://blog.csdn.net/biggbang)

> 
> ? yc 也是一样的用法
> 
> 
> 

---
* **针对括号** ( ( ), < >, [ ], { } )

这一部分用来处理**括号括起来的文本**。

> 
> ? 注意,**这里的操作是支持多行的!!**
> 
> 
> 

> 
> i (inner) 在这里的意思是操作的部分**不包括**两侧的括号
> a (around) 在这里的意思是操作的部分**包括**两侧的括号
> 
> 
> 

di(di) 剪切光标所在的**小括号( )括起来的**文本(i代表不包括两侧的括号) ?

[![diBracket-2022-04-12](https://img-blog.csdnimg.cn/img_convert/800d9c8b777e4563942d1c69ee32c20c.gif)](https://blog.csdn.net/biggbang)

da{da} 剪切光标所在的**大括号{ }括起来的**文本(a代表包括两侧的括号) ?

[![daBracket-2022-04-12](https://img-blog.csdnimg.cn/img_convert/74f6e16bc9556011c89e9f2fa94aef42.gif)](https://blog.csdn.net/biggbang)

> 
> ? yc 也是一样的用法,适用于多种括号。
> 
> 
>[[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2jkTHoZp-1650734010052)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/quoteAndBracketComic-2022-04-12.png)]](https://blog.csdn.net/biggbang)
  1. 双引号指定寄存器

默认情况下使用复制粘贴似乎用的都是一个公共的寄存器(Registers),Vim实际上是支持寄存多条文本的,只需要在指令前指定寄存器名即可。

* **默认的寄存器**被系统记为 " ,一个双引号

当执行yy时,会将目标文本行存入 " 寄存器,操作等同于:

""yy

> 
> ? 其中第一个单引号代表**引用寄存器**,第二个单引号便是**默认寄存器的名字**。
> 
> 
> 

于是粘贴指令也可以改写成:""p,即从默认寄存器中取出文本进行粘贴。

---
* **指定寄存器名字**

把上述第二个单引号改成 a-z 的**小写字母**即可!这样就相当于得到了26个“剪贴板”呢~

比如将当前行复制到a寄存器中:

"ayy

那么粘贴该内容的指令也显而易见了:

"ap

---
* **往已命名寄存器中附加内容**

上面写到用**小写字母**对寄存器进行命名,那么**大写字母**呢?

大写字母用于往寄存器中**附加内容**:

"Ay$ 将光标位置到行尾的文本附加到a寄存器中 ?

[![AppendToRegisterA-2022-04-12](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/AppendToRegisterA-2022-04-12.gif)](https://blog.csdn.net/biggbang)这里主要记录一下复制粘贴方面的寄存器引用方法。Vim寄存器储存的内容远不止这么一点,还包括操作历史记录等信息,就不多赘述了。

?记忆方法:" 就像个图钉?,标明是哪一个寄存器。

末行模式/命令行模式下可以查看寄存器信息:reg[isters]

  1. 缩进

Vi/Vim 常用于编辑代码,缩进功能当然也是比较常用的了。

* **增加缩进**

>> 给**光标所在行**增加缩进
4>> 给**光标所在行**以及下面3行增加缩进,共4行
* **减少缩进**

<< 给**光标所在行**减少缩进
5<< 给**光标所在行**以及下面4行减少缩进,共5

?记忆方法:> 可以看作在往右边“顶”文本,而 < 可以看作在往左边“拉”文本。

  1. 撤销与重做

文本编辑必不可少的便是撤销与重做功能了!

* **撤销**

u 撤销**上一步操作**

大写 U 撤销**光标所在行**的**所有操作**

> 
> ?记忆方法:u直接记成 undo [v]撤销,取消 即可。
> 一定要辨别小写u和大写U的操作区别!
> 
> 
>
* **重做**

Ctrl + r 或者 Ctrl + R

重做操作和小写u的操作是反向的!重新执行撤销掉的操作。

> 
> ?记忆方法:r直接记成 redo [v]重做 即可。
> 为什么多了个Ctrl呢?因为r已经被**替换模式**占用了!
> 
> 
>
  1. 数字快捷增减

要修改一个数值的时候可以进入插入模式,移动光标到数值上进行修改,这样可能略嫌麻烦了。在命令模式/普通模式下提供了对数值进行递增递减的操作指令:

* Ctrl + aCtrl + A亦可)

对光标指针所在数值进行**递增操作**。如果光标没有指向数值,会**向右查找**数值进行递增。

> 
> ? 记忆方法: a 可以记成 ascend [v]上升;升高;递增,
> 亦可以记成 accumulate [v]积累;逐渐增加。
> 之所以有 Ctrl ,是因为a已经被用于切换**插入模式**了!
> 
> 
> 

[![ascendingNum-2022-04-12](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/ascendingNum-2022-04-12.gif)](https://blog.csdn.net/biggbang)

? 示例中可以看到,当光标右方没有数值时会屏幕闪烁提示指令无效。

---
* Ctrl + xCtrl + X亦可)

对光标指针所在数值进行**递减操作**。如果光标没有指向数值,同样会**向右查找**数值进行递减。

> 
> ? 记忆方法: x 可以记成 叉掉数字,从而让数值减小。
> 之所以有Ctrl也是因为x已经被用于**剪切字符**。
> 
> 
> 

---
* **重复指令**

设想一下,如果我要增加10亿,难不成要一直按住按键不动?当然不是这样啦!

再记忆一遍,命令模式/普通模式下的指令可以加**数字前缀**进行重复,在这里只需要这样输入指令即可快速递增指定数值:

1000000000Ctrl + a

[![ascendBillionTimes-2022-04-12](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/ascendBillionTimes-2022-04-12.gif)](https://blog.csdn.net/biggbang)

Ctrl + x 同理!

[![rocketAscending-2022-04-12](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/rocketAscending-2022-04-12.png)](https://blog.csdn.net/biggbang)
* **进制的表示方法**

0x 前缀表示**十六进制** (记**hex**的**x**)

0 前缀表示**八进制** (记**octal**的**o**,虽然这里是**0**)

0b0B 前缀表示 **二进制**

---

? Vim支持进行增减的类型有:二进制数(binary),八进制数(octal),十进制数(decimal,默认支持),十六进制数(hexadecimal),无符号数(unsigned),字母(alphabetical)。

? 具体配置见末行模式/命令(行)模式nrformats配置一节。

  1. 字母大小写切换

数字增减有了,字母其实也是可以“随地大小变”的!

* gu后接光标移动指令

将从光标位置开始到**执行移动指令后**光标的位置结束作为选区,
将选区内的所有字母转换为**小写**!

[![guiw-2022-04-16](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/guiw-2022-04-16.gif)](https://blog.csdn.net/biggbang)

? 演示了guiw指令。iw选择的是光标所在单词FLOOD,执行gu后将单词所有字母转换为**小写**:flood。
* gU后接光标移动指令

将选区内的所有字母转换为**大写**

? 记忆方法:gugU中一个是小写u,一个是大写U。大写U可以记成UPPERCASE 大写,而当U小写成为u,对应的便是小写了。g仍然可以记成gogo UPPERCASE

* ~ (波浪符)

波浪符将**光标**所在的字母进行**大小写交换**,比如b会转换为BB则会转换为b。

这个指令可以用于转换**光标所在行**中**自光标往后**的字母**大小写**。

[![tilde-2022-04-16](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/tilde-2022-04-16.gif)](https://blog.csdn.net/biggbang)

? 50~,将本行中自光标往后50个字符进行大小写转换。
* g~后接光标移动指令

上面的~只能对**光标所在行**的字符进行大小写交换,如果要涉及**一个选区**呢?

于是,g~他来了!在g~后接光标移动指令,将**光标当前位置**开始到**执行指令后光标所在位置**为止之间的字符进行大小写交换。

[![gT2gj-2022-04-17](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/gT2gj-2022-04-17.gif)](https://blog.csdn.net/biggbang)

? g~2gj,将从光标位置开始到光标向下移动**两个屏幕行**为止的部分字符大小写交换。

? 记忆方法:波浪线可以看作是一条,随着波的传播,同一个地方的波峰波谷交替出现,在这里就记成大写小写的交替

1739f851f001553e9df33ff68dbdf769 - 想法子记忆Vi/Vim常用操作及指令

  1. 重复最后一次更改

.

↑ 没错,指令就是一个句点!

比如,之前我输入了3dd剪切了三行文本,之后再按一次 . 会再往下剪切三行;

又比如,我先输入了x剪切光标所指的字符,之后输入 3. 就会再剪切三个字符。

但是!,如果我先输入3j让光标往下移动3行,再按 . 是没有效果的,因为j指令仅仅是移动了光标,没有对文本造成任何更改!

⚠ 注意! . 重复的是最后一次更改,而不是最后一次指令!

? 记忆方法:硬背

  1. 行连接指令
* 大写 J

这个指令能将**光标所在行**和其下面一行**连接成一行**,之间用**一个空格间隔**:

[![LinesJoin-2022-04-17](https://img-blog.csdnimg.cn/img_convert/5e966e1099cb2a402fae5bb1427850ed.gif)](https://blog.csdn.net/biggbang)

> 
> 4J 的话则是将光标所在行和**下面3行**进行连接,每行之间用空格隔开。**一共4行**。
> 
> 
> 

> 
> ? 记忆方法:直接记 Join [v]加入;连接 即可。
> 
> 
>
* gJ

同样是连接行。和上面J不同的是,这个指令在连接行之后**不会在之间添加空格!**。

> 
> ? 记忆方法:可以记成gross Joingross总的;粗鲁的的意思,gross Join就可以记成 “**连接成一个总体**” 或者 “**粗鲁地直接连在一起**”。
> 
> 
>

末行模式/命令(行)模式

在这个模式中涉及到了最基础的文件保存 / 退出编辑器等操作。

  1. :w 写入 (记为Write [v]写文件

顾名思义,这个指令会将自开始编辑以来的更改写入文件中:

21689751236c4b297f8ecfa2263b5cc8 - 想法子记忆Vi/Vim常用操作及指令

aae4c5f0622426c785ba4eeddbbb47e2 - 想法子记忆Vi/Vim常用操作及指令

文件保存总让人想起桌面软件的“另存为”功能,当然Vi也是能实现的:

:w filename

ebe8998c66f4ec10fdc09f70627c970c - 想法子记忆Vi/Vim常用操作及指令

f616ead908ff768462900c5c65efa7b1 - 想法子记忆Vi/Vim常用操作及指令

除了上面针对单一文件的写入指令,还可以同时写入多个文件,下面这个指令适用于Vim编辑器中同时打开多个文件的情况:

:wa

? 将所有打开的文件进行写入保存。(可以记成Write All 写入所有

9a66df762c2323551c8b58c396cf4c91 - 想法子记忆Vi/Vim常用操作及指令

如果只想保存文本中的一部分呢?Vim还支持指定行写入文件:

:, w filename

? 比如我想写入第13至第14行的文本(共两行)::13,14 w filename

c4e3a8f090c0484bccb08c81baf7e633 - 想法子记忆Vi/Vim常用操作及指令

输出文件效果:

6b3f5db52d95c6e1d5382f30cd4de89c - 想法子记忆Vi/Vim常用操作及指令

  1. :q 退出(记为Quit [v]离开;关闭

文本编辑器必备操作——关闭文件。

:q 指令会退出当前编辑的文件(如果只打开了一个文件,这个操作同时会退出编辑器

:qa 指令则会退出所有打开的文件,并且退出编辑器

? 记忆方法:Quit All 关闭所有

注意:上面这两个指令执行的前提是当前文件自打开后没有任何更改,也就是说文件缓冲区中没有任何新增内容,不然编辑器会警告文件尚未保存更改。
怎么强制退出呢?下面就将写到.....

  1. 联合使用 wq

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-syhCaFkR-1650734010060)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/wCombineWithQ-2022-04-18.png)]

既然未保存没法正常关闭文件,那么保存后关闭不就行了嘛!

:wq

? 双剑合璧,数据无忧!这个指令会将缓冲区中的编辑内容写入文件,并且关闭文件。(针对当前正在编辑的文件

当然,也有针对编辑器所有打开的文件的操作:

:wqa

? 写入并关闭所有打开的文件,这之后会退出编辑器。(Write & Quit All

  1. 感叹号 ! 强制执行指令

有些情况下末行模式指令的执行可能会被拒绝,比如上面的:q在文件保存前不能直接执行。(这一点其实也是为了数据安全)

⭐ 然而,只需要在这些指令末尾加上感叹号 ! 就可以强制执行了!

比如未保存强制退出编辑器就可以用: :qa!

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gC8RYmqR-1650734010061)(https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/forcedQuitAll-remake-2022-04-18.png)]

此外我觉得文件强制写入也是值得记录一下的::w!

? 这个指令会尝试强制写入文件,即使文件是只读(Read-only)的,前提是用户得要有修改文件属性的权限

⚠ 强制执行指令时一定要再三考虑,防止数据丢失等问题。

  1. :r 在下一行读入文件 (记成 Read [v]读取

在编辑文本的时候可能需要将另一个文件的文本插入进来,这个时候就轮到:r大显身手了。

:r 能够读取一个文件,并将文件内容追加到光标所在行的后一行

b36613a257bc8219c50ea3df8e798ad0 - 想法子记忆Vi/Vim常用操作及指令

? 左边是待读取的文件,右边是正在编辑的文件。利用:r指令,成功将文件内容插入到光标后一行。

  1. :e 开启新buffer或刷新当前buffer (记成 Edit [v]编辑

当编辑文件编辑到一半,想要推翻当前的所有修改,该怎么办呢?

一种方法是使用 :q!,强制退出文件;

另一种方法则是 :e!,强制刷新当前buffer

* 文件在Vi/Vim中未经修改但是在**别的软件中受到了修改**时,可以直接使用 :e 在编辑器里**刷新(重新载入)文件内容**。
* 文件做了修改后,也可以用 :e! 强制**重载文件内容**,也就是抛弃文件开启以来的所有修改。

[![forcedRefreshBuffer-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/forcedRefreshBuffer-2022-04-22.gif)](https://blog.csdn.net/biggbang)

buffer可以理解为文件的一个缓冲区,文件在编辑器中被编辑实际上也就是临时buffer在被编辑。当写入文件的时候就是把buffer中的内容写入到原文件中,以达成编辑的效果。


除了刷新当前文件的buffer外,:e指令还可以用于打开新的文件并创建buffer。

* :e  打开文件(如果存在的话),创建buffer

? 前提:目前在编辑的文件已经保存或者未经修改。
* :e!  无论**当前文件是否保存(写入)**,强制打开新文件
  1. :buffers:bn, :bp 以及 :b

    • :buffers

    列出所有buffer,在这里可以看到buffer的ID(最左边一列)。

    edb5d523c2a2f125c21d205aa1c5078b - 想法子记忆Vi/Vim常用操作及指令

    • :b

    既然拿到了buffer的ID,我就可以在buffer之间跳转了,比如:

    :b2 跳转到id为2的buffer

    注意:这个操作也要求当前编辑的文件不能有未保存的修改。

    • :bn:bp

    就算不知道buffer的ID,利用这两个指令也能在buffer间互相切换:

    :bn ---> 切换到下一个buffer

    :bp ---> 切换到上一个buffer

    注意:这个操作也要求当前编辑的文件不能有未保存的修改。

    ? 记忆方法:bnbuffer next 下一个bufferbpbuffer previous 上一个buffer

  2. :help [command]:h [command] 查看帮助

就像Linux中的man指令一样,Vi/Vim的末行模式/命令行模式也有个:help指令用来查看编辑器使用帮助。

⭐ 直接使用 :help 时会打开帮助文件,并将光标移动到顶部。

⭐ 指令形如:help [command]时,在开启文件后光标会跳转到对应的指令帮助信息。

比如要看寄存器指令的帮助信息::help reg:help registers:h reg 当然也行 )

  1. :registers:reg 查看寄存器

这一部分可以结合命令模式/正常模式双引号指定寄存器一起记。

⭐ 直接使用 :registers:reg 时会展示所有的寄存器 ?

cb9b003fdde965964619cc95c22b6d35 - 想法子记忆Vi/Vim常用操作及指令

也可以在后面指定要查询的寄存器名,这些名称可以用空格分开,也可以连在一起写:

⭐ 比如 :reg[isters] a b c:reg[isters] abc ,可以筛选查询a,b,c寄存器 ?

19795a3d5639ea64ff843045c2de22b5 - 想法子记忆Vi/Vim常用操作及指令

关于寄存器更多用法这里就不多赘述了~

860100270a97240ba8a3cadc9a64c457 - 想法子记忆Vi/Vim常用操作及指令

  1. :marks 查看标记信息

这个和上面:reg[isters]的使用很类似,所以咱给写一起了。

这一部分可以结合命令模式/正常模式自定义跳转标记一起记。

⭐ 直接使用 :marks 能查询到所有现有的标记

和:registers一样,在后接查询的标记名,能筛选结果:

⭐ 比如 :marks a b c:marks abc ,可以筛选查询a,b,c标记

  1. :set 设置项

这个指令用于设置一些配置项,这里就记几个比较常用的(因为配置项有非常多,全写出来太累赘了)

* **开启/关闭行号显示**

:set number:set nu 可以在行首显示行号

而 :set nonumber:set nonu 则可以关闭行号显示

此外 :set nu!:set number! 可以在**开启行号/关闭行号**之间互相切换。

[![lineNumbers-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/lineNumbers-2022-04-19.gif)](https://blog.csdn.net/biggbang)

> 
> ? 记忆方法:直接记 Number [n]号码 [v]标号
> 
> 
>
* **开启/关闭相对行号显示**

:set relativenumber:set rnu 可以显示相对行号

而 :set nonumber:set nonu 则可以关闭相对行号显示

此外 :set rnu!:set relativenumber! 可以在**开启相对行号/关闭相对行号**之间互相切换。

[![relativeNumber-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/relativeNumber-2022-04-19.gif)](https://blog.csdn.net/biggbang)

? 可以看到,相对行号以**光标所在行**为第0行,向上或向下标出相对的行号。

这个特性可以用于**辅助涉及多行的指令**,比如 y3j, 3k, 2>> 等。

> 
> ? 记忆方法:按字面记,即 Relative Number 相对 号码
> 
> 
>
* **识别数字进制的配置**

上面已经写过Ctrl+xCtrl+a能[快捷减增数字](https://blog.csdn.net/biggbang)。不过呢,Vim怎么识别**增减的类型**呢? 这就涉及到下面这个配置项了:

    + :set nrformats:set nf

    直接输入这个指令会查询**该配置项当前的设置**:

    [![queryNrformats-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/queryNrformats-2022-04-19.gif)](https://blog.csdn.net/biggbang)

    上面也写过Vim[能识别的增减类型](https://blog.csdn.net/biggbang)。这里的octal,hex 则代表会尝试识别**八进制**,**十六进制**以及**十进制**的数字。(十进制是默认支持的)
    + 使用 +=-= 进行配置

    这种写法其实可以理解为一些编程语言中的**运算赋值**操作:

    :set nrformats+=bin

    :set nrformats+=bin,alpha

    (或者 set nf+= ... )

    ? 多个可以用**逗号**分隔,会“附加”到配置项的末尾。

    然而,移除的时候就要注意了,一定要按**配置中的顺序**来:

    :set nrformats-=bin

    :set nrformats-=bin,alpha

    ? 这样写的话必须保证配置项中也有**bin,alpha**这个字串!

    [![removeNf-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/removeNf-2022-04-19.gif)](https://blog.csdn.net/biggbang)

    ↑ 没有bin,alpha这个字串,无法移除。

    [![successfullySetNf-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/successfullySetNf-2022-04-19.gif)](https://blog.csdn.net/biggbang)

    ↑ 移除octal,alpha的话就很成功,因为配置项中有这个字串。
    + 直接使用 = 进行配置

    同样可以按编程语言里的**赋值**来理解,会直接**改变整个配置**

    :set nrformats=octal,hex

    ? 多个值一样是用**逗号分隔**
> 
> ? 记忆方法:nrformats可以看成number recognizing formats,也就是数字识别格式。取nf即为nf,因此nf也可以简单记为number format!
> 
> 
>
  1. 执行Shell指令

编辑文本到一半突然想执行一个Shell指令。我之前可能会利用:wq先退出编辑器,输入执行指令后再重新打开编辑器。

实际上在末行模式/命令行模式下Vi/Vim也是能快捷切换到Shell执行指令的:

* :!  **暂时离开**编辑界面,在Shell下执行指令**并打印结果**

[![temporaryShell-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/temporaryShell-2022-04-19.gif)](https://blog.csdn.net/biggbang)

? **临时**跳转到命令行执行清屏查询日期指令并展示结果。

> 
> ? :! 还有很多神奇的用法,比如执行指令后将返回的结果插入下一行::r! ;
> 又比如将文本2125行进行升序排序,并替换原文本::21,25! sort,这里就很像Shell中管道符的用法了。
> 
> 
> 

[![sortByShell-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/sortByShell-2022-04-19.gif)](https://blog.csdn.net/biggbang)

? 演示 :21,25! sort

> 
> 更多用法在这里就不多赘述了╮(╯3╰)╭
> 
> 
>
* :sh **创建一个新的Shell会话**

这个指令就比较简单粗暴了,在执行后会创建一个新的Shell会话,我可以直接在Shell中执行指令!

[![createNewShell-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/createNewShell-2022-04-19.gif)](https://blog.csdn.net/biggbang)

这个Shell是一个**非登入**Shell,所以需要使用 exit 指令退出。

该Shell退出后会**回到编辑器界面!**

[![exitShell-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/exitShell-2022-04-19.gif)](https://blog.csdn.net/biggbang)

> 
> ? 这个直接记忆 Shell 就行
> 
> 
>
  1. 重复上一条指令

命令模式/正常模式下我可以使用 . 句点来重复上一次更改。在末行模式/命令行模式下也有类似的方法。

* **经典方法**

在末行模式/命令行模式下输入 : 后会进入末行模式/命令行模式,此时可以通过:

⭐ , , PageUp, PageDown

来浏览之前的输入历史

[![scrollHistory-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/scrollHistory-2022-04-19.gif)](https://blog.csdn.net/biggbang)

? 可以在 : 后面输入一些字符以**加快检索**:

[![scrollHistoryWithHint-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/scrollHistoryWithHint-2022-04-19.gif)](https://blog.csdn.net/biggbang)

? 输入:s后,能快速浏览:s开头的历史指令记录;:se则能快速浏览:se开头的历史指令。
* **@: 方法**

在命令模式/正常模式下输入:

@:

能重新执行**上一条末行模式/命令行模式指令**。

> 
> ? 记忆方法:把 : 末行指令给**at**出来!
> 
> 
> 

虽然我把这个方法写在末行模式这里了,但实际上其可以算是命令模式/正常模式的指令。

因此也是**可以重复执行的!**

[![repeatLastLine-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/repeatLastLine-2022-04-19.gif)](https://blog.csdn.net/biggbang)

? 演示:先用:d删除一行,然后再用3@:重复执行:d三次以继续删除下面三行。[![however-2022-04-19](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/however-2022-04-19.png)](https://blog.csdn.net/biggbang)

有点累了...喝口水...

基本搜索替换

  1. 搜索

前提:在命令模式/正常模式下:

/<搜索模式> 自光标向后查找

?<搜索模式> 自光标向前查找

? 其实末行模式/命令行模式下也不是不行,也就多了个冒号::/<搜索模式>:?<搜索模式>。不过接下来这两个指令就不行了 ↓

n ---> 这个指令会重复上一次搜索动作

* 比如/....是往**光标后**搜索,按下n就是**在光标后**再次搜索
* 而?....是往**光标前**搜索,那么按下n就是**在光标前**再次搜索N ---> 这个指令和 n 的操作相反

? 记忆方法:n可以记成 next [a]下一个的 ,正如o和O操作相反一样,Nn操作相反。


这里的搜索模式实际上就是正则表达式,不过和普通的正则略微有些不同!

1. 当然是可以直接搜索字串的

[![searchStr-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/searchStr-2022-04-21.gif)](https://blog.csdn.net/biggbang)
2. **大小写不敏感**搜索

一般在写正则表达式时要进行大小写不敏感匹配我一般会在正则表达式末尾加上i标记,但是在搜索模式里是不行的,需要用到特殊转义标记:

\c

这个标记写在搜索模式的中间(不要写在中括号[]里!)一般也是可以的,但我觉着还是**放在末尾**好辨别一些。

[![caseInsensitivelySearch-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/caseInsensitivelySearch-2022-04-21.gif)](https://blog.csdn.net/biggbang)

[![caseInsensitiveSearch-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/caseInsensitiveSearch-2022-04-21.png)](https://blog.csdn.net/biggbang)

> 
> ? 记忆方法:这里的 \c 可以记成 case [n]大小写。
> 
> 
>
3. 利用**正则表达式**进行搜索

这里使用正则表达式和在编程语言里有些小区别,这里简单写一下:

    * 采用 <> 分别匹配**单词词首**和**单词词尾**,而不是\b。
    * 部分元字符**要发挥元字符**的作用的话,需要**先转义**。这里列个表格:

    | 元字符 | 用途 | 在搜索模式中的用法 |
    | --- | --- | --- |
    | + | 匹配1次或多次 | \+ |
    | ? | 匹配0或1次(非贪婪模式在Vim中另有元字符) | \? |
    | {n,m} | 匹配n到m次 | \{n,m} |
    | {n,} | 匹配n次或更多次 | \{n,} |
    | {n} | 匹配n次 | \{n} |
    | {,m} | 匹配0到m次 | \{,m} |
    | < | 匹配单词首部 | \< |
    | > | 匹配单词尾部 | \> |
    | ( | 子模式(捕获组)开始标志 | \( |
    | ) | 子模式(捕获组)结束标志 | \) |
    | | | 两项之间任意匹配一个 | \| |
    | % | 用于修饰**非捕获组** | \% |
    | @<=, @, @=, @!` | 向后预查,向前预查 | \@, \@, \@=, \@!` |

    > 
    > ? 这种情况是基于Vim**默认设置**的magic模式的,其他模式不多赘述。
    > 默认情况下这些元字符如果不转义,就代表**匹配这个字符本身**。
    > 
    > 
    > 

    除此之外的元字符大多是可以直接使用的,下面是一些示例:

    [![regexSearch1-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/regexSearch1-2022-04-21.jpg)](https://blog.csdn.net/biggbang) [![regexSearch2-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/regexSearch2-2022-04-21.jpg)](https://blog.csdn.net/biggbang) [![regexSearch3-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/regexSearch3-2022-04-21.png)](https://blog.csdn.net/biggbang) [![regexSearch4-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/regexSearch4-2022-04-21.jpg)](https://blog.csdn.net/biggbang) [![regexSearch5-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/regexSearch5-2022-04-21.jpg)](https://blog.csdn.net/biggbang)

    另外也是支持**引用子模式的匹配结果(分组)**的:

    [![regexSearch6-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/regexSearch6-2022-04-22.jpg)](https://blog.csdn.net/biggbang)
4. 神奇的**非贪婪模式**

Vim这里的非贪婪模式用的就不是元字符 ? 了,取而代之借用了一下大括号 {...,...} 

> 
> 官方说明可以在末行模式/命令行模式下输入:help non-greedy查看。
> 
> 
> 

当 { 后面**紧接**了一个连字符(Hyphen) - 时,就相当于采用了**非贪婪匹配**,下面举些例子:

e\{2,3} 匹配 eeeee,如果有更长的eee就优先匹配(也就是**贪婪匹配**)

e\{-2,3} 匹配 eeeee,优先匹配更短的ee(也就是**非贪婪匹配**)

[![non-greedyMatch-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/non-greedyMatch-2022-04-21.gif)](https://blog.csdn.net/biggbang)

> 
> ? 这个例子中在{后紧接-后会往少的匹配,也就是只匹配一个paprika 。
> 
> 
> 

---

如果在 - 后面**没有指定匹配次数**,就有点类似 * 了。形如:

\{-} 匹配0-任意次,但是非贪婪匹配,匹配次数尽量少。

\{-,3} 匹配0-3次,但是非贪婪匹配,匹配次数尽量少。

> 
> 例:用模式 ke\{-} 匹配字串keep只匹配到了k,因为\{-}代表匹配0-任意次,但是**非贪婪匹配**,所以这里e匹配了0次。
> 
> 
>
5. **非捕获组与向前/向后预查**

在编程语言使用的正则表达式中,非捕获组一般用(?:模式)来表示,这一个子模式不会参与分组。

---

然鹅在Vi/Vim**搜索模式**中,非捕获组用的是:

\%(子模式\)\%\(子模式\) 也行)

来表示的。下面就是一个示例:

[![nonCapturingGroup-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/nonCapturingGroup-2022-04-22.jpg)](https://blog.csdn.net/biggbang)

> 
> 可以看到accept并未参与分组,\1引用的是第二个子模式的匹配结果after
> 
> 
> 

---

接着,在编程语言中的向后/向前预查(?<=), (?=), (?, (?!) 在Vi/Vim中也是有特殊的表示方法的:

| 预查内容↓ \ 方向→ | 向后(左) | 向前(右) |
| --- | --- | --- |
| 存在 | \(子模式\)\@<= | \(子模式\)\@= |
| 不存在 | \(子模式\)\@ | \(子模式\)\@! |

> 
> ? **预查**即预先检查子模式,看看这个子模式是否存在于待匹配的字串前/后,预查子模式是**不会出现在最终的匹配结果中的**。
> 
> 
> 

[![lookBehindPositive-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/lookBehindPositive-2022-04-22.jpg)](https://blog.csdn.net/biggbang)

> 
> ? 匹配 paprika,前提是 paprika**后面**(左边)必须要匹配到paprika。
> 
> 
> 

[![lookAheadPositive-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/lookAheadPositive-2022-04-22.jpg)](https://blog.csdn.net/biggbang)

> 
> ? 匹配paprika,前提是paprika**前面**(右边)必须要匹配到 pepper。
> 
> 
> 

[![lookAheadNegative-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/lookAheadNegative-2022-04-22.jpg)](https://blog.csdn.net/biggbang)

> 
> ? 匹配paprika,前提是paprika**前面**(右边)**不能**匹配到pepper。
> 
> 
>
6. 搜索**光标下的单词**

这一小节的操作是在命令模式/正常模式下的:

    * * **往后**搜索**光标目前指向的单词**,只匹配**一整个单词**
    * # **向前**搜索**光标目前指向的单词**,只匹配**一整个单词**

    [![searchWordUnderCursor-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/searchWordUnderCursor-2022-04-21.gif)](https://blog.csdn.net/biggbang)

    > 
    > 从这个例子可以看到,实际上Vim是把光标指向的单词转换成了搜索语句。* 对应 /\# 对应 ?\<\accept\>。
    > 
    > 
    >
    * g* **往后**搜索**光标目前指向的单词**,单词可作为**字串的一部分**被匹配。
    * g# **向前**搜索**光标目前指向的单词**,单词可作为**字串的一部分**被匹配。

    [![searchWordPartUnderCursor-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/searchWordPartUnderCursor-2022-04-21.gif)](https://blog.csdn.net/biggbang)

    > 
    > 可以看到 * 对应 /accept# 对应 ?accept,单词可作为**字串的一部分**被匹配。
    > 
    > 
    >
> 
> ? 因为这几个指令被转换为末行搜索操作了,所以在搜索中可以用的nN这一类指令也是可以用的。
> 
> 
>
7. 开启**搜索高光显示**

上面的图示中搜索匹配项都会“**黄的发光**”,这种**匹配结果高光显示**是可以作为配置项使用 :set 进行设置的:

    * :set hlsearch:set hls 开启搜索结果**高光显示**
    * :set nohlsearch:set nohls 关闭搜索结果**高光显示**

    [![highlightSearch-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/highlightSearch-2022-04-21.gif)](https://blog.csdn.net/biggbang)虽然高亮显示一目了然,但是开了之后光标就不明显了,高光咱已经看够了,怎么**关掉目前结果的高光展示**呢?

    * :nohlsearch:noh 关闭**目前的高光显示**

    [![nohlsearch-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/nohlsearch-2022-04-21.gif)](https://blog.csdn.net/biggbang)

    > 
    > 可以看到输入:noh后会取消目前的高亮,但是这并不影响重新开始搜索时高亮展示匹配。
    > 
    > 
    >
> 
> ? 记忆方法:hlsearchHighlight SearchHighlight [v]突出,强调Search [n]搜索。
> 
> 
> 

[![hlsearch-2022-04-21](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/hlsearch-2022-04-21.png)](https://blog.csdn.net/biggbang)
  1. 替换

替换的指令就是在末行模式/命令行模式下输入的了,

其完全可以结合搜索指令一起记忆:

:[作用范围] s/<搜索模式>/<替换模式>/<替换flag>

? 很明显了,这里的s代表的就是substitute [v]取代,替代

1. **作用范围**

这里直接列表直观展示一下:

| 作用范围 | 说明 |
| --- | --- |
| 不写 | **默认光标所在行** |
| . | 同样代表**光标所在行** |
| n (整数) | 代表**第n行**。比如233代表第233行 |
| $ | 代表**最后一行** |
| 'a | 代表a**标记所在行**(关于设定跳转标记可以看[自定义跳转标记](https://blog.csdn.net/biggbang)) |
| 上述符号±相对行数 | 比如'a+2就代表a标记所在行的下面第二行;.-1 就代表光标所在行的上面一行 |
| 上述符号1,上述符号2 | 代表从上述符号1代表的行到上述符号2代表的行。比如.,$代表从光标所在行到最后一行。 |
| % | 代表**从第一行到最后一行**(整个文本),和1,$相同 |
| '<,'> | 在[**可视模式**](https://blog.csdn.net/biggbang)下输入 : 进入末行模式/命令行模式会自动填充,代表**可视模式下选定的范围 |

[![replaceInTheRange-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/replaceInTheRange-2022-04-22.gif)](https://blog.csdn.net/biggbang)

> 
> ? 示例:.,+3 代表从光标所在行到光标所在行的下面第三行作为范围进行替换
> 
> 
>
2. **搜索模式**

同上面的[**搜索模式**](https://blog.csdn.net/biggbang)。

? 值得一提的是这里可以留空。如果留空的话**默认使用**上一次[**搜索**](https://blog.csdn.net/biggbang)的匹配。

> 
> 比如我先搜索了/\,然后我想替换全文的/\/pepper,在替换时可以这样写::%s//pepper/g(省略了搜索模式)。
> 
> 
>
3. **替换模式**

替换模式当然可以是**任意字符串**,

同时也可以**引用搜索结果中的部分**:

[![replaceWithReference-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/replaceWithReference-2022-04-22.gif)](https://blog.csdn.net/biggbang)

? 使用 \1, \2 引用**搜索结果**中的第一个和第二个**捕获组**的内容。

[![replaceWithReference2-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/replaceWithReference2-2022-04-22.gif)](https://blog.csdn.net/biggbang)

? 使用 & 引用**搜索匹配结果**

[![replaceWithReference3-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/replaceWithReference3-2022-04-22.gif)](https://blog.csdn.net/biggbang)

? 很巧妙的用法,利用捕获组引用交换顺序
4. **替换flag**

一般正则表达式末尾会加上修饰符以控制匹配行为。而Vim这里的**替换flag**也是类似修饰符,控制替换行为的。

下面列出几个比较常用的:

| Flag | 说明 | 记忆 |
| --- | --- | --- |
| g | 全局替换(在指定的**作用范围**内) | global [a]全局的 |
| i | 忽略大小写 | ignore-case 忽略大小写 |
| I | 大小写敏感 | 和i意义相反 |
| n | 并不会真正替换,而是显示**匹配的数目** | numerate [v]计算 |
| c | 在替换前**让用户确认** | confirm [v]确认 |

> 
> ? 值得一提的是 i——忽略大小写。
> 之前在[**搜索模式**](https://blog.csdn.net/biggbang)中提到过在模式中加入 \c 能忽略大小写进行搜索,而这里替换也是用的搜索模式,
> 因此在这里就算不用修饰符i,转而在搜索模式中加入\c也是完全可以的。
> 
> 
> 

[![replaceIgnoreCase-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/replaceIgnoreCase-2022-04-22.gif)](https://blog.csdn.net/biggbang)

> 
> ? 不使用修饰符而是在搜索模式中使用了\c
> 
> 
> 

---

[![numerateMatches-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/numerateMatches-2022-04-22.gif)](https://blog.csdn.net/biggbang)

> 
> ? 使用 n 修饰符,并不会实际替换文本,而是**显示匹配的数量**
> 
> 
> 

---

[![confirmReplace-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/confirmReplace-2022-04-22.gif)](https://blog.csdn.net/biggbang)

> 
> ? 使用 c 修饰符,在替换前**让用户确认**。
> 
> 
> 

其中提示输入的 (y/n/a/q/l/^E/^Y) 中,分别代表:

| 输入指令 | 说明 | 记忆 |
| --- | --- | --- |
| y | 确认替换**当前光标所在匹配** | yes [v]确认 |
| n | 不替换**当前匹配** | no [v]不 |
| a | 确认替换**所有匹配** | all [v]所有 |
| q | 不进行替换并退出 | quit [v]退出 |
| l | 替换**当前匹配**并退出 | instead [adv]顶替 |

(另还有Ctrl+ECtrl+Y,是光标移动指令)

除了这些,搜索替换可能还有一些其他的用法尚未提到,另外还有:g这种末行指令。这里就主要记录一下基础的用法啦~

871f5a3d2fa36c16cec092a6f0dc324c - 想法子记忆Vi/Vim常用操作及指令

简单多窗口编辑

一个文件一个文件地打开编辑老累了,幸运的是,Vim支持多窗口编辑。

  • 打开新窗口

    1. :new [filename]

    使用:new指令能在编辑器顶部新建一个横向窗口(原本的文件窗口会下移

    如果有指定文件名,则会在新窗口打开指定文件(文件不存在就是单纯的创建buffer)

    2d724eff70554078487546502c010d66 - 想法子记忆Vi/Vim常用操作及指令

    1. :sp[lit] [filename] (也就是:sp:split都可以)

    这个指令行为上和:new挺类似的,都会新建一个横向窗口,使原本的窗口下移

    不同的是,没有指定文件名时会在新窗口打开当前编辑的文件

    7ad18cb66194bf7e0df938ba12c3c366 - 想法子记忆Vi/Vim常用操作及指令

    ? split [v]划分,分割

    1. :vs[plit] [filename]

    这个指令和:sp[lit]类似,但是:vs[plit]是新建一个纵向窗口,使原本的窗口右移

    :sp[lit]一样的是,没有指定文件名时会在新窗口打开当前编辑的文件

    2d3ed20cc3abc914d44a0adc99f6c37a - 想法子记忆Vi/Vim常用操作及指令

    ? :vsplitvertical split [v]垂直分割

  • 在窗口之间切换

开了这么多窗口,得想个法子来回切换哈!

1. Ctrl+wCtrl+W ( 可以记成 Control Window 指挥窗口 )

连按两次 Ctrl+w/W 可以在**窗口之间循环切换**

[![switchWindow-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/switchWindow-2022-04-22.gif)](https://blog.csdn.net/biggbang)
2. Ctrl+w/W+方向控制键

对于**分割成上下关系**的窗口,可以使用 Ctrl+w/W+↑/↓Ctrl+w/W+k/j 来进行上下切换

[![switchSplitWindow-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/switchSplitWindow-2022-04-22.gif)](https://blog.csdn.net/biggbang)

而对于**分割成左右关系**的窗口,就可以用 Ctrl+w/W+←/→Ctrl+w/W+h/l 来进行左右切换

[![switchVSplitWindow-2022-04-22](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/switchVSplitWindow-2022-04-22.gif)](https://blog.csdn.net/biggbang)
  • 调整窗口尺寸

在调整终端尺寸的时候,Vim里的这些窗口可能并不会如意调整展示尺寸。这里记录一些改变窗口尺寸的指令:

1. **调整窗口高度**

:res[ize] [±][height]

> 
> :res[ize]指令有两种调整**窗口高度**的方法。
> 
> 
> 

第一种是设置**绝对高度**,比如:res 5,将窗口高度设置为5行:

[![resizeWindow-2022-04-23](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/resizeWindow-2022-04-23.gif)](https://blog.csdn.net/biggbang)

第二种是设置**相对高度**,比如:res +5,将窗口高度增加5行:

[![resizeWindowRelatively-2022-04-23](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/resizeWindowRelatively-2022-04-23.gif)](https://blog.csdn.net/biggbang)
2. **调整窗口宽度**

:vert[ical] res[ize] [±][width]:res[ize]类似,:vert[ical] res[ize]也是能相对/绝对地调整**窗口宽度**的:

[![vresizeWindow-2022-04-23](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/vresizeWindow-2022-04-23.gif)](https://blog.csdn.net/biggbang)

? 这个记起来可能有点反直觉,
vertical resize 垂直方向上调整,可以记作此时只有垂直高度固定的窗口,所以调整的是宽度
而普通的resize则是调整的窗口高度,可以记作此时只有水平宽度是固定的窗口

  • 退出窗口

末行模式/命令行模式下的 :q 退出

可视模式

前面的笔记有几个地方涉及到了可视模式

可视模式下的操作可谓是简化版的命令模式/正常模式了,显而易见,“简化”的就是“选择文本”的操作了。


在高光选择了文本后可以进行以下操作:

  • 拷剪粘三件套

48d6119972a52d2bb294a850af1ab4fc - 想法子记忆Vi/Vim常用操作及指令

这里和命令模式/正常模式的不同之处就在于文本已经选择完毕,只需要输入一个指令就可以完成简单的编辑操作。

+ y 输入一次即可拷贝**选择的文本**,而Y则是拷贝**所有选择文本所在的行** (Yank)
+ d 输入一次即可剪切**选择的文本**,D也是会剪切**所有选择文本所在的行**的。(Delete)
+ xd操作一致 (给文本画叉叉)
+ cd操作一样,不过剪切完之后会**进入插入模式**
+ p 在选择的区域中**粘贴文本** ( Paste
  • 替换

r<字符> 将所选文本替换为<字符>Replace )

ed4a9813140d3e60e5bf5d1ec55a47a1 - 想法子记忆Vi/Vim常用操作及指令

? ra将所选块文本全部替换为了a

  • 行连接

这一部分和命令模式/正常模式下的行连接指令几乎是一样了。

+ J 将所选文本**对应的行**连接成一行,**用空格间隔** ( Join )

[![spaceJoinedRow-2022-04-23](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/spaceJoinedRow-2022-04-23.gif)](https://blog.csdn.net/biggbang)
+ gJ 将所选文本**对应的行**连接成一行,**没有间隔**

[![directlyJoinedRow-2022-04-23](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/directlyJoinedRow-2022-04-23.gif)](https://blog.csdn.net/biggbang)
  • 执行末行模式/命令行模式指令

在可视模式下选择文本后输入 : 进入末行模式,会自动填充成 :'<,'>,这代表在所选文字中进行末行语句操作

之前的搜索替换中就有提到过:基本搜索替换->作用范围

  • 代码缩进

命令模式/正常模式下要对一行代码进行缩进操作,往往需要输入两次><

而在可视模式下只需要输入一次:

5c03986aa8134a16a9cbe6a24f53989b - 想法子记忆Vi/Vim常用操作及指令

? > 将所选文本增加一次缩进

83cb2f983c31b41c339a2cc3b4620a05 - 想法子记忆Vi/Vim常用操作及指令

? < 将所选文本减少一次缩进(因为选择不完全,只减少了一部分)

0f7626f8ef6c2017e6d9bace98cc3156 - 想法子记忆Vi/Vim常用操作及指令

? > 将所选文本增加2次缩进

  • 大小写转换

这一部分和命令模式/正常模式下的大小写转换指令是很类似的:

+ u 将所选文本转换为**小写**
+ U 将所选文本转换为**大写** ( UPPERCASE )
+ ~ 将所选文本中的**大写**和**小写**分别转换为**小写**和**大写**[![caseSwitchInVisualMode-2022-04-23](https://cdn.jsdelivr.net/gh/cat-note/bottleassets@latest/img/caseSwitchInVisualMode-2022-04-23.gif)](https://blog.csdn.net/biggbang)

? 演示了一下使用~进行大小写转换

可视模式还有很多用法,这里就不再写更多了(再写更多就成语法手册了喂!)

In the end

回过头来...我又写了这么长一篇笔记...稍后我会完善一下目录功能以增加阅读体验。

这篇笔记虽然写得多,但是涉及的内容实际上还是比较浅层的,毕竟这篇笔记主要目的是为了记忆常用方法。也就是说,Vi/Vim的使用技巧远不止这么一点,更多高级的操作还需日常多加使用才能更加熟练!

因个人能力极其有限,可能文章中会出现部分错误。如果有写错的地方,也请大家多指教!

最后...感谢你读到这里,再会~

74b1f830836d925a444b6a8a2147c6f4 - 想法子记忆Vi/Vim常用操作及指令

.floatCatalogue {
position: fixed;
transform: translateX(100%);
display: none;
overflow: auto;
z-index: 5000;
right: 0;
top: 0;
width: 40%;
height: 100%;
background-color: rgba(255, 255, 255, 0.9);
transition: .5s ease;
}

.floatCatalogue::-webkit-scrollbar {
display: none;
}

.floatCatalogue .head {
position: relative;
height: 3em;
}

.floatCatalogue span {
position: fixed;
display: block;
color: #484848;
font-size: 1.5em;
font-weight: bolder;
margin: .5em;
}

.floatCatalogue .content {
margin: .5em;
}

.floatCatalogue .close {
display: block;
position: fixed!important;
right: 0;
top: 0;
font-size: 2em!important;
z-index: 5001;
color: #3f3f3f!important;
cursor: pointer;
margin: .2em .3em!important;
background-color:transparent!important;
border:none!important;
transition: .5s ease;
}

.floatCatalogue .close:hover {
color: #888888!important;
text-decoration: underline!important;
}

.floatCatalogue p {
margin: .6em 10px;
word-break: keep-all;
}

.floatCatalogue p a {
color: #6a6a6a!important;
text-decoration: none!important;
transition: .5s ease;
}

.floatCatalogue p a:hover {
color: #484848!important;
text-decoration: underline!important;
}

@media screen and (max-width: 820px) {
.floatCatalogue {
width: 100%;
}
}

目录

关于Vi/Vim名词本身#

编辑器模式#

模式切换#

命令模式

插入模式

i/I 插入

a/A 附加

o/O 新增一行

s/S 剪切(取代)

切换指令的重复

替换模式

末行模式/命令行模式

可视模式

命令模式/正常模式#

模式切换

基本光标移动(上下左右)

h l k j移动

↑ ↓ ← →移动

其他款式

行内光标移动

跳到行首

跳到行末

屏幕可见行中移动光标

g^ g0 gm g$ 等等

H M L 指令

文件内光标移动

按词移动w,b

移动到词尾e

移动到句首( )

移动到段首{ }

跳转光标gg,G

自定义跳转标记

设置标记m

跳转到标记处`

跳转到标记所在行的行首'

删除标记delm

翻页

往回翻

往前翻

复制y

复制一行

复制多行

光标移动复制

光标移动复制总结

剪切d,c,x...

剪切一行

剪切多行

剪切一个字符x

行内剪切D,C

光标移动剪切

光标移动剪切总结

粘贴p

粘贴行文本

粘贴不足一行文本

Vim上下文编辑

针对单词diw,daw...

针对句子yis,das...

针对段落yip,dap...

针对单/双引号

针对括号

双引号指定寄存器

默认寄存器

命名寄存器

往命名寄存器中附加内容

缩进

增加缩进

减少缩进

撤销与重做

撤销

重做

数字快捷增减

数字递增Ctrl+A/a

数字递减Ctrl+X/x

重复递增/递减操作

进制的表示方法

Vim可识别类型

字母大小写切换

转换为小写gu

转换为大写gU

大小写交换~

光标移动大小写交换g~

重复最后一次更改.

行连接

带空格间隔

无间隔

末行模式/命令(行)模式#

写入(保存)文件:w

另写入文件(另存为)

写入所有文件

写入文件部分行

退出文件:q

联合使用:wq

强制执行!

在下一行读入文件:r

:e开启或刷新当前buffer

强制重载文件内容

简述buffer

打开新文件

:buffers, :bn, :bp, :b

查看所有:buffers

:b跳转指定buffer

buffers之间互相切换:bn,:bp

查看帮助:h[elp]

查看寄存器:reg[isters]

查看标记:marks

设置项:set

开启/关闭行号显示

开启/关闭相对行号显示

识别数字进制的配置

查看当前的配置

运算赋值配置±=

赋值配置=

执行Shell指令

暂时离开编辑执行语句

创建新的Shell会话

重复上一条指令

经典方法

@:方法

基本搜索替换#

搜索

搜索模式(Pattern)

直接搜索字串

大小写不敏感搜索

利用正则表达式

非贪婪匹配

非捕获组与向前/向后预查

搜索光标下单词

开启搜索匹配高光展示

替换

作用范围

搜索模式

替换模式

替换标识(Flag)

简单多窗口编辑#

打开新窗口

:new

:sp[lit]

:vs[plit]

在窗口之间切换

Ctrl+w/W循环切换

Ctrl+w+方向键切换

调整窗口尺寸

调整高度

调整宽度

退出窗口

可视模式#

拷剪粘三件套y,d,p...

替换r

行连接J,gJ

执行末行语句

调整代码缩进

大小写转换u,U,~

In the end#

转载请注明:xuhss » 想法子记忆Vi/Vim常用操作及指令

喜欢 (0)

您必须 登录 才能发表评论!