| 快捷键 | Change | Delete | Yank(Copy) |
| ------ | ------ | ------ | ---------- |
| Line | cc | dd | yy |
| Letter | cl | dl | yl |
| Word | cw | dw | yw |
## 跳转
### 字符移动
h,j,k,l 左下上右
### 行内移动
```bash
w 正向移动到相邻单词的首字符
b 逆向移动到相邻单词的首字符
B 向前一个单词, 以空格和 TAB 为分隔符
e 正向移动到相邻单词的尾字符
ge 逆向移动到相邻单词的尾字符
0 数字0,左移光标到本行的开始;
$ 右移光标,到本行的末尾;
^ 移动光标,到本行的第一个非空字符
### 页移动
```bash
H 跳转到当前屏幕的第一行
M 跳转到本屏显示的中间一行
L 跳转到最后一行
2H 表示将光标移到屏幕的第2行
3L 表示将光标移到屏幕的倒数第 3 行
z + enter 是当前行成为屏幕的第一行
z + - 是当前一行成为最后一行
ctrl + f 在文件中前移一页(相当于 page down);
ctrl + b 在文件中后移一页(相当于 page up);
ctrl + d 往下滚动半屏
ctrl + u 往上滚动半屏
### 文件移动
```bash
gg 跳到首行
G 调到尾行
nG 跳转到 n 行
% 跳到另一边括号
### 配合查找字符的方式移动
```bash
fa 正向移动到第一个字符 a 处
Fa 逆向移动到第一个字符 a 处
### 非相邻的单词或字符间移动
```bash
8w 正向移动到相隔八个单词的首字符
4Fa 逆向移动到第四个 a 字符
### 更大范围的移动
```bash
当光标停留在一个单词上,* 键会在文件内搜索该单词,并跳转到下一处;
\#当光标停留在一个单词上,# 在文件内搜索该单词,并跳转到上一处;
(/) 移动到 前/后 句 的开始;
{/} 跳转到 当前/下一个 段落 的开始。
g_ 到本行最后一个不是 blank 字符的位置。
fa 到下一个为 a 的字符处,你也可以 fs 到下一个为 s 的字符。
t, 到逗号前的第一个字符。逗号可以变成其它字符。
3fa 在当前行查找第三个出现的 a。
F/T 和 f 和 t 一样,只不过是相反方向;
gg 将光标定位到文件第一行起始位置;
G 将光标定位到文件最后一行起始位置;
NG 或 Ngg 将光标定位到第 N 行的起始位置
## 搜索匹配
```bash
/text 向后搜索
? text 向前搜索
: g/targetWord 全局搜索
:%s/oldWord/newWord/gc 全局替换
n 搜索下一个同样的内容
N 搜索上一个同样的内容
## 替换和删除
Vim 常规的删除命令是 d、 x (前者删除 行 ,后者删除 字符 ), 结合 Vim 的其他特性可以实现基础的删除功能。将光标定位于文件内指定位置后,可以用其他字符来替换光标所指向的字符,或从当前光标位置删除一个或多个字符或一行、多行
```bash
gg dG 删除全部内容
d0 删除至行首
dl 删除当前字符, dl = x
dh 删除前一个字符
dd 删除当前行
dj 删除上一行
dk 删除下一行
dw 删除到下一个单词开头
de 删除到本单词末尾
dE 删除到本单词末尾包括标点在内
db 删除到前一个单词
dB 删除到前一个单词包括标点在内
10d 删除当前行开始的 10 行
d$ 删除当前字符之后的所有字符(本行)
D 删除当前字符至行尾。D = d$
kd gg 删除当前行之前所有行(不包括当前行)
jdG 删除当前行之后所有行(不包括当前行)
: 1, 10d 删除 1-10 行
: 11, $d 删除 11 行及以后所有的行
: 1, $d 删除所有行
J 删除两行之间的空行,实际上是合并两行
rc 用 c 替换光标所指向的当前字符;
nrc 用 c 替换光标所指向的前 n 个字符;
5rA 用 A 替换光标所指向的前 5 个字符;
x 删除光标所指向的当前字符;
nx 删除光标所指向的前 n 个字符;
3x 删除光标所指向的前 3 个字符;
dw 删除光标右侧的字;
ndw 删除光标右侧的 n 个字;
3dw 删除光标右侧的 3 个字;
db 删除光标左侧的字;
ndb 删除光标左侧的 n 个字;
5db 删除光标左侧的 5 个字;
dd 删除光标所在行,并去除空隙;
ndd 删除(剪切) n 行内容,并去除空隙;
3dd 删除(剪切) 3 行内容,并去除空隙;
Vim 常规的替换命令有 c 和 s ,结合 Vim 的其他特性可以实现基础的替换功能,不过替换命令执行以后,通常会由 编辑模式 进入 插入模式
s 用输入的正文替换光标所指向的字符;
S 删除当前行,并进入编辑模式;
ns 用输入的正文替换光标右侧 n 个字符;
nS 删除当前行在内的 n 行,并进入编辑模式;
cw 用输入的正文替换光标右侧的字;
cW 用输入的正文替换从光标到行尾的所有字符(同 c$ );
ncw 用输入的正文替换光标右侧的 n 个字;
cb 用输入的正文替换光标左侧的字;
ncb 用输入的正文替换光标左侧的 n 个字;
cd 用输入的正文替换光标的所在行;
ncd 用输入的正文替换光标下面的 n 行;
c$ 用输入的正文替换从光标开始到本行末尾的所有字符;
c0 用输入的正文替换从本行开头到光标的所有字符。
## 复制粘贴
```bash
yy 拷贝当前行
nyy 拷贝当前后开始的 n 行,比如 2yy 拷贝当前行及其下一行
*“+y 复制 1 行到操作系统的粘贴板
*“+nyy 复制 n 行到操作系统的粘贴板
: 1, 10 co 20 将 1-10 行插入到第 20 行之后
: 1, coco 将整个文件复制一份并添加到文件尾部
正常模式下按 v(逐字)或 V(逐行)进入可视模式,然后用 jklh 命令移动即可选择某些行或字符,再按 y 即可复制
ddp 交换当前行和其下一行
xp 交换当前字符和其后一个字符
正常模式下按 v(逐字)或 V(逐行)进入可视模式,然后用 jklh 命令移动即可选择某些行或字符,再按 d 即可剪切
ndd 剪切当前行之后的 n 行。利用 p 命令可以对剪切的内容进行粘贴
: 1, 10d 将 1-10 行剪切。利用 p 命令可将剪切后的内容进行粘贴。
: 1, 10 m 20 将第 1-10 行移动到第 20 行之后。
x 剪切当前字符
3x 剪切当前光标开始向后三个字符
X 剪切当前字符的前一个字符。X = dh
p 粘贴到下一行或右侧,修改和删除的数据自动到粘贴板中
P 粘贴到上一行或左侧
如果是要替换别的单词,则先按 v 进入 visual mode,选中要替换的单词,再按粘贴即可, 粘贴板中就换成了被替换的单词
yiw (yank inner word)在一个单词的任意字母使用, 就复制该单词
yw 则只是复制从光标所在字母到词尾的部分
## 撤销和重复
```bash
u 撤销最近一次修改 undo
. 重复最后一条修改正文的命令
U 撤销所有修改
ctrl + r 取消最后一次的撤销 redo
## 块编辑
```bash
v 可进入 visual 模式,使用标准快捷键移动光标可选择文本块,之后可输入标准编辑命令
ctrl + v 列编辑
## 命令行模式下的一些技巧
```bash
DTc 删除从光标的 c 之间的所有字符
Rc 将光标的字符替换为 c
nDD 删除 n 行数据
nYY 复制 n 行数据
nX 删除 n 个字符
R 进入替换状态,esc 退出
## 插入模式
### 进入
```bash
i 在当前位置生前插入
I 在当前行首插入
a 在当前位置后插入
A 在当前行尾插入
o 在当前行之后插入一行
O 在当前行之前插入一行
s 删掉当前字符,并进行输入
x 删掉当前字符,停留在 Normal 模式
### 退出
```bash
ctrl + O 暂时性的
ctrl + C 取消当前的任何操作
ctrl + [ 官方推荐替换 Esc
### 移动光标 尽量不要进入插入模式移动光标
```bash
ctrl + H 光标移当前行行首 imap
ctrl + J 光标移下一行行首 imap
ctrl + K 光标移上一行行尾 imap
ctrl + L 光标移当前行行尾 imap
Alt + H 光标左移一格 imap
Alt + J 光标下移一格 imap
Alt + K 光标上移一格 imap
Alt + L 光标右移一格 imap
## 命令模式
### 打开、保存
```bash
: e path_to_file/filename 在已经启动的 Vim 中打开一个文件
: w 保存当前编辑的文件
: w file_temp 将当前文件另存为 file_temp
### 退出
```bash
ZZ 保存并退出
: wq 保存并退出
: e! 放弃所有修改,并打开原来文件
Shift +Z, Q 无条件退出
q! 无条件退出
ctrl + Z 退出 vim,不推荐,会生成.swp 的文件
### 行号与文件
**编辑中的每一行正文都有自己的行号,用下列命令可以移动光标到指定行**(效果与 编辑模式 下的 ngg 或 nG 相同)
```bash
:n 将光标移到第 n 行
**命令模式下,可以规定命令操作的行号范围**。数值用来指定绝对行号;字符“.”表示光标所在行的行号;字符符“$”表示正文最后一行的行号;简单的表达式,例如“.+5”表示当前行往下的第 5 行。例如
```bash
:345 将光标移到第 345 行
:345w file 将第 345 行写入 file 文件
:3,5w file 将第 3 行至第 5 行写入 file 文件
:1,.w file 将第 1 行至当前行写入 file 文件
:.,$w file 将当前行至最后一行写入 file 文件
:.,.+5w file 从当前行开始将 6 行内容写入 file 文件
:1,$w file 将所有内容写入 file 文件,相当于 :w file 命令
**在命令模式下,允许从文件中读取正文,或将正文写入文件**
```bash
:w 将编辑的内容写入原始文件,用来保存编辑的中间结果
:wq 将编辑的内容写入原始文件并退出编辑程序(相当于 ZZ 命令)
:w file 将编辑的内容写入 file 文件,保持原有文件的内容不变
:a,bw file 将第 a 行至第 b 行的内容写入 file 文件
:r file 读取 file 文件的内容,插入当前光标所在行的后面
:e file 编辑新文件 file 代替原有内容
:f file 将当前文件重命名为 file
:f 打印当前文件名称和状态,如文件的行数、光标所在的行号等
### 字符串搜索
在 编辑模式 讲过字符串的搜索,此处的 命令模式 也可以进行字符串搜索,给出一个字符串,可以通过搜索该字符串到达指定行。如果希望进行正向搜索,将待搜索的字符串置于两个 / 之间;如果希望反向搜索,则将字符串放在两个 ? 之间
```bash
:/str/ 正向搜索,将光标移到下一个包含字符串 str 的行
:?str? 反向搜索,将光标移到上一个包含字符串 str 的行
:/str/w file 正向搜索,并将第一个包含字符串 str 的行写入 file 文件
:/str1/,/str2/w file 正向搜索,并将包含字符串 str1 的行至包含字符串 str2 的行写
## Vim 中的正则表达式
```bash
:/struct/ 要搜索一行正文,这行正文的开头包含 struct 字
因为它只找出在行中任意位置包含 struct的第一行,并不一定在行的开始包含 struct 。解决问题的办法是在搜索字符串前面加上特殊字符^
:/^struct/
也可以用类似办法在搜索字符串后面加上表示行的末尾的特殊字符 $ 来找出位于行末尾的字
:/struct$/
下表给出大多数特殊字符和它们的含义
```bash
^ 放在字符串前面,匹配行首的字;
$ 放在字符串后面,匹配行尾的字;
\< 匹配一个字的字头;
\> 匹配一个字的字尾;
. 匹配任何单个正文字符;
[str] 匹配 str 中的任何单个字符;
[^str] 匹配任何不在 str 中的单个字符;
[a-b] 匹配 a 到 b 之间的任一字符;
* 匹配前一个字符的 0 次或多次出现;
\ 转义后面的字符。
## 正文替换
> 利用 : s 命令可以实现字符串的替换
```bash
:%s/str1/str2/ 用字符串 str2 替换行中首次出现的字符串 str1
:s/str1/str2/g 用字符串 str2 替换行中所有出现的字符串 str1
:.,$ s/str1/str2/g 用字符串 str2 替换正文当前行到末尾所有出现的字符串 str1
:1,$ s/str1/str2/g 用字符串 str2 替换正文中所有出现的字符串 str1
:g/str1/s//str2/g 功能同上
:m,ns/str1/str2/g 将从m行到n行的str1替换成str2
从上述替换命令可以看到:
g 放在命令末尾,表示对搜索字符串的每次出现进行替换, 不止匹配每行中的第一次出现;不加 g,表示只对搜索字符串的首次出现进行替换;g 放在命令开头,表示对正文中所有包含搜索字符串的行进行替换操作
s 表示后面跟着一串替换的命令
% 表示替换范围是所有行,即全文
## 统计当前文件中字符串 str1 出现的次数
```bash
:%s/str1/&/gn
## 删除正文
Vim 的初级删除命令是用 d ,高级删除命令可以用 正则替换 的方式执行
```bash
:d 删除光标所在行
:3d 删除 3 行
:.,$d 删除当前行至正文的末尾
:/str1/,/str2/d 删除从字符串 str1 到 str2 的所有行
:g/^\(.*\)$\n\1$/d 删除连续相同的行,保留最后一行
:g/\%(^\1$\n\)\@<=\(.*\)$/d 删除连续相同的行,保留最开始一行
:g/^\s*$\n\s*$/d 删除连续多个空行,只保留一行空行
:5,20s/^#//g 删除5到20行开头的 # 注释
## 恢复文件
Vim 在编辑某个文件时,会另外生成一个临时文件,这个文件的名称通常以 . 开头,并以 .swp 结尾。Vim 在正常退出时,该文件被删除,若意外退出,而没有保存文件的最新修改内容,则可以使用恢复命令 : recover 来恢复文件,也可以在启动 Vim 时用 -r 选项
## 选项设置
为控制不同的编辑功能,Vim 提供了很多内部选项。利用 : set 命令可以设置选项。基本语法为
```bash
:set option 设置选项 option
常见的功能选项包括:
```bash
autoindent 设置该选项,则正文自动缩进
ignorecase 设置该选项,则忽略规则表达式中大小写字母的区别
number 设置该选项,则显示正文行号
ruler 设置该选项,则在屏幕底部显示光标所在行、列的位置
tabstop 设置按 Tab 键跳过的空格数。例如 :set tabstop=n,n 默认值为 8
mk 将选项保存在当前目录的 .exrc 文件中
## 分屏
```bash
: vsplit(可用缩写 : vsp) 左右分屏
: split(可用缩写 : sp) 上下分屏
ctrl + w + hjkl 窗口之间移动
ctrl + w + w 逆时针遍历
ctrl + w = 让所有的屏都有一样的高度;
ctrl + w + 增加高度;
ctrl + w - 减少高度。
另外,也可以在终端里启动 vim 时就开启分屏操作
vim -On file1 file2… 打开 file1 和 file2 ,垂直分屏
vim -on file1 file2… 打开 file1 和 file2 ,水平分屏
## 标签页
Vim 的标签(Tab)页,类似浏览器的标签页,一个标签页打开一个 Vim 的窗口,一个 Vim 的窗口可以支持 N 个分屏
```bash
:tabnew 在Vim中新建一个标签
:tabnew filename 如果要在新建标签页的同时打开一个文件,则可以在命令后面直接附带文件路径
Vim 中的每个标签页有一个唯一的数字序号,第一个标签页的序号是 0,从左向右依次加一。关于标签页有一系列操作命令,简介如下
```bash
:tN[ext] 跳转到上一个匹配的标签
:tabN[ext] 跳到上一个标签页
:tabc[lose] 关闭当前标签页
:tabdo 为每个标签页执行命令
:tabe[dit] 在新标签页里编辑文件
:tabf[ind] 寻找 'path' 里的文件,在新标签页里编辑之
:tabfir[st] 转到第一个标签页
:tabl[ast] 转到最后一个标签页
:tabm[ove] N 把标签页移到序号为N位置
:tabnew [filename] 在新标签页里编辑文件
:tabn[ext] 转到下一个标签页
:tabo[nly] 关闭所有除了当前标签页以外的所有标签页
:tabp[revious] 转到前一个标签页
:tabr[ewind] 转到第一个标签页
## 外部工具集成
Vim 可以与许多外部程序集成,功能十分强大,比如 diff , ctags , sort , xxd 等等
## diff
```bash
Linux 命令 diff 用来对比两个文件的内容,不过对比结果显示在终端里,可读性比较差。结合 Vim,在终端里可以直接输入命令 vimdiff,后面跟两个文件名作为参数:
vimdiff file1 file2
即可在 Vim 里分屏显示两个文件内容的对比结果,对文件内容差异部分进行高亮标记,还可以同步滚动两个文件内容,更可以实时修改文件内容,方便程度和用户体验大大提高。
vimdiff a.txt b.txt
如果直接给 -d 选项是一样的
vim -d a.txt b.txt
除了在终端里开启 vimdiff 功能,也可以在打开 Vim 后,在 Vim 的命令模式输入相关命令来开启 vimdiff 功能:
: diffsplit abc.txt
如果你现在已经开启了一个文件,想 Vim 帮你区分你的文件跟 abc.txt 有什么区别,可以在 Vim 中用 diffsplit 的方式打开第二个文件,这个时 候 Vim 会用 split(分上下两屏)的方式开启第二个文件,并且通过颜色,fold 来显示两个文件的区别
这样 Vim 就会用颜色帮你区分开 2 个文件的区别。如果文件比较大(源码)重复的部分会帮你折叠起来。
: diffpatch filename
通过 : diffpatch 你的 patch 的文件名,就可以以当前文件加上你的 patch 来显示。vim 会 split 一个新的屏,显示 patch 后的信息并且用颜色标明区别。
如果不喜欢上下对比,喜欢左右(比较符合视觉)可以在前面加 vert ,例如:
: vert diffsplit abc.txt
: vert diffpatch abc.txt
看完 diff,用 : only 回到原本编辑的文件,觉得 diff 的讨厌颜色还是在哪里,只要用 : diffoff 关闭就好了。
还有个常用的 diff 中的就是 : diffu , 这个是 : diffupdate 的简写,更新的时候用
## sort
Linux 命令 sort 可以对文本内容进行按行中的字符比较、排序,但在终端里使用 sort 命令处理文件,并不能实时查看文件内容。具体用法请自查手册。
## xxd
vim+xxd 是 Linux 下最常用的二进制文本编辑工具,xxd 其实是 Vim 外部的一个转换程序,随 Vim 一起发布,在 Vim 里调用它来编辑二进制文本非常方便。
首先以二进制模式在终端里打开一个文件:
vim -b filename
Vim 的 -b 选项是告诉 Vim 打开的是一个二进制文件,不指定的话,会在后面加上 0x0a ,即一个换行符。
然后在 Vim 的命令模式下键入:
:%! xxd
即可看到二进制模式显示出来的文本,看起来像这样:
```bash
0000000: 1f8b 0808 39d7 173b 0203 7474 002b 4e49 ....9..;..tt.+NI
0000010: 4b2c 8660 eb9c ecac c462 eb94 345e 2e30 K,......b..4^.0
0000020: 373b 2731 0b22 0ca6 c1a2 d669 1035 39d9 7;'1.".....i.59
然后就可以在二进制模式下编辑该文件,编辑后保存,然后用下面命令从二进制模式转换到普通模式:
```bash
:%!xxd -r
另外,也可以调整二进制的显示模式,默认是 2 个字节为一组,可以通过 g 参数调整每组字节数:
```bash
:%!xxd -g 1 表示每1个字节为1组
:%!xxd -g 2 表示每2个字节为1组(默认)
:%!xxd -g 4 表示每4个字节为1组
Home Page