0. 序言
编程人员花费最长时间的开发工具可能就是编辑器了,一个非常方便、高效的编辑器对开发人员来说是非常有效的。在unix/linux下,甚至windows下,vim都可以说是个非常优秀的编辑器。虽然许多朋友开发过程中都在使用vim,但通常只使用了vim非常有限的功能,没有接触到vim的很多非常符合开发人员需求的功能。下面就从开发人员的角度出发,介绍开发过程经常使用到的功能。读完本文,您会发现vim原来是如此强大的集成开发环境。
Vim是基于GPL的开源项目,是对vi的提升版本,而vi是unix环境下最通用的编辑器。不过由于vim的强大的功能,许多最新发布的linux版本中已经把vim作为默认vi编辑器了。如果您的系统中没有安装vim,可以从vim.sourceforge.net上下载合适的版本。闲话少提,下面从基本编辑命令、通用编辑命令、开发常用命令等方面介绍vim。本文以介绍功能为主,使用细节多尝试一下就明白了。以下讨论基于vim 6.2。
1.基本编辑命令
本节简单分类基本的vim编辑命令,只给出简单的功能说明,作为使用vim的命令速查,详细描述请查看vim的在线帮助。Vim下输入:h <命令>即可。下面的命令是在常态模式下(就是任何状态下多按几次ESC键后到达的状态,为了避免概念模糊,这里对这些状态不进行描述)输入的字符序列。
命令 |
含义 |
命令 |
含义 |
移动 |
|||
h j k l |
左下上右(和方向键相同含义) |
H M L |
同一页(屏幕)内的第一行、中间行、 最后一行 |
ctrl+f ctrl+b |
下一页 上一页 |
ctrl+d ctrl+u |
下半页 上半页 |
b w e |
前一个单词首 后一个单词首 后一个单词尾 |
0 ^ $ |
行首 行首的非空字符 行尾 |
|
|
|
|
输入 |
|||
a A |
当前字符后输入 行尾输入 |
i I |
当前字符前输入 行前输入 |
o O |
当前行下插入新行输入 当前行上插入新行输入 |
:r [filename] |
当前行下插入文件[filename]的内容 |
|
|
|
|
改变 |
|||
d[范围] |
删除范围内的文字,如:dw(删除到词尾)d$(删除到行尾) |
c[范围] |
改变,删除指定范围后直接进入编辑状态 |
~[范围] |
改变当前字符或者范围内的大小写 |
J |
连接当前行和下一行 |
r[char] |
替换字符,使用随后输入的字符替换当前字符 |
R |
进入替换模式,输入的字符替换文本的文字 |
y[范围] |
拷贝范围内的文字 |
p |
粘贴到当前字符后 |
x |
删除当前字符 |
. |
重复最后一条改变文本内容的命令 |
|
|
|
|
查找 |
|||
f[char] F[char] |
行内向后查找字符[char] 行内向前查找字符[char] |
t[char] T[char] |
行内向后查找字符[char]前 行内向前查找字符[char]前 |
/[string] ?[string] |
向下查找[string] 向上查找[string] 查找后使用n继续查找,使用N向相反方向继续查找 |
* # |
向下查找光标当前单词 向上查找光标当前单词 |
n N |
重复最后一次/ ? 命令 相反方向重复最后一次/ ? 命令 |
|
|
|
|
|
|
其他 |
|||
:e filename |
关闭当前文件,编辑文件 |
:w filename |
写入文件中 |
ctrl+g :g |
显示文件信息 |
:help [cmd] |
查看vim的命令cmd的帮助信息 |
:u |
undo一个改变 |
:red |
重做undo的改变 |
U |
undo当前行所有的改变 |
|
|
注:
[范围] 是界定上个动作的范围,vim里面称为motion。上面的改变类命令都属于motion命令,意思是指移动了光标位置。
命令 |
含义 |
命令 |
含义 |
0 |
到行第一个字符 |
^ |
到行第一个非空字符 |
$ |
到行最后一个字符 |
|
|
[n]k [n]j |
向上几行 向下几行 |
[n]w |
删除n个单词 |
|
|
|
|
上面的分类是个人为便于记忆进行分类,没有特定标准或已有规则。
2. 通用命令
2.1. 搜索
搜索应该说是开发过程中非常常用的命令。最常见的搜索命令是/ 和? ,后面加需搜索的词句或者正则表达式(简单的介绍见下面一小结)。然后使用n 或者N 找下一个或者上一个搜索结果。
开发过程中最可能的情况是看到一个词希望能够找到它,这时候* (向下搜索当前词)和#(向上搜索当前词)命令就非常方便。如果想搜索包含当前词的词都可以可以使用g* 或者g# 命令。
找到变量的定义,可以使用gd(当前词的局部变量定义)和gD(当前词的全局变量定义)命令,这两个命令不一定非常准确,但有时候还是比较有用的。
当前词的宏定义,可以使用[_ctrl-d 找到当前词第一个出现在#define 后的位置。而[d 显示当前文件和包含文件中当前词的第一个宏定义,[D 列出所有的宏定义。] 开始的命令是从当前位置开始搜索。
在当前文件和包含头文件中进行搜索,搜索宏定义可以使用上面说的命令,函数、变量等的声明、引用的搜索,可以使用[i (从文件头开始搜索包含当前词的位置)命令,[I 列出所有的行。] 开始的命令是从当前位置开始搜索。Vim的命令还是比较有一致性的。跳到出现的位置则使用[_ctrl-i (从第一行开始),或者]_ctrl_i (从当前位置开始)。
多文件的搜索可以通过外部命令:grep 来实现,缺省情况和使用grep命令非常类似,查找出来的列表,可以使用Quickfix类命令来在列表中循环访问,通过此类命令也可以在多个grep命令列表中进行跳转,此类命令初始为编译出错时设计的,在3.4节介绍相关命令时详细描述。
替换命令格式比较长,但有时候又非常有用,必须理解它每一个细节才能保障灵活使用。命令格式如下:
:[range]s[ubstitute]/{pattern}/{string}/[&][c][e][g][p][r][i][I] [count]
这个命令的含义就是在range的范围内,查找pattern定义的模式替换为string。
Range的定义可以参考:help range,通常可以指定行数,标记,甚至指定一个搜索也可以,例如:
3,.+1 从3行到当前行的后一行
‘t,$-2 从标记t到文章尾后两行
% 全部文章,相当于.,$
.,/{pattern}/ 从当前行到下一个匹配pattern的行
pattern和上面介绍的搜索模式一样可以是一个字符串或者是正则表达式。
String通常可以是字符串,其他特殊内容可以参考:help sub-replace-special 。
最后的标记的含义如下:
& 保持上个替换命令的标记
c 每次替换进行确认
g 对每行所有匹配的模式进行替换,缺省一行只替换一次
i 忽略大小写
I 不忽略大小写
2.2. 正则表达式
使用正则表达式极大的增强了搜索、替换的描述能力,但这也增加了学习的难度。这里把开发过程中可能使用的标记简单的罗列一下,然后举几个例子,大家应该可以举一反三,大致知道如何使用了。匹配模式的学习可以参考:help pattern 。
命令 |
含义 |
命令 |
含义 |
ordinary atom |
|||
^ \_^ |
匹配行首,必须在pattern开始 任何地方匹配行首 |
$ \_$ |
匹配行尾,必须在pattern最后 任何地方匹配行尾 |
. \_. |
匹配任意字符,不能在行尾 任意地方匹配字符 |
\< \> |
词首 词尾 |
|
|
|
|
multi items |
|||
* |
0或多个 |
\+ |
1或多个 |
\? |
0或1个 |
\{n,m} |
n到m个,尽量匹配多数个 |
\{-n,m} |
n到m个,尽量匹配少数个 |
|
|
character classes |
|||
\s \S |
空白字符<space> <TAB> 非空白字符 |
\d \D |
数字 非数字 |
\x \X |
十六进制数字 非十六进制数字 |
\a \A |
字母a-zA-Z |
\w \W |
词字符包括字母、数字和下划线 |
\h \H |
首字符包括字母和下划线 |
|
|
|
|
其他 |
|||
\e \t \r \b \n |
<ESC> <TAB> <CR> <BS> <NewLine> |
\c \C |
忽略大小写 匹配大小写 |
|
|
|
|
这样c语言的标志符就可以使用 \<\h\w* 来匹配;
\<ca\w*5 匹配以ca为词开始中间包含5的任意词;
2.3. 多窗口
开发过程中有时候需要打开多个文件或者看一个文件的不同部分,这种情况就是多窗口的功用做在。
打开一个新窗口使用 :split [filename] 命令,加文件名打开的窗口编辑此文件,否则编辑当前文件。
关闭当前窗口使用 :q 即可。
Ctrl-w j 下一个窗口
Ctrl-w k 上一个窗口
ctrl-w = 所有窗口等高、宽
ctrl-w + 增加当前窗口高度
ctrl-w – 减少当前窗口高度
不同的文件在内存中的数据是一个buffer,一个buffer可以有1个或多个window,甚至没有window。在多个文件间跳转(后面介绍)时,通常文件所在buffer仍然在内存。可以使用 :buffers 列出当前内存中所有buffer。
使用:bn (下一个buffer)和 :bp (上一个buffer)在buffer间循环。
2.4. 可视模式
图形界面使用太多了,可能觉得使用命令界定范围不直观,可视模式则允许可视的选择一块文本,然后输入相关命令对其进行操作。其使用过程是移动到块开始字符,输入命令进入可视模式;使用各种移动命令移动到块尾;输入相关命令对此块内容进行操作。
v 进入字符选择可视模式
V 进入行选择可视模式
Ctrl-v 进入块选择可视模式
2.5. 标记(mark)
对某行做一个标记有时候是非常有用的功能。以后可以方便的移动到此行,在motion中也可以引用这个位置。
设置标记使用 m{a-zA-Z} 设置一个标记,名字是后面跟的字符,一个文件内可以定义a-z共26个标记,文件间可以定义一个A-Z共26个标记。
移动到标记行,使用‘[char] 或者 `[char] 命令移动到[char]对应的标记。
2.6. 文件恢复
敲半天的东西不能因为断电丢失了。Vim缺省输入200个字符或者空闲4秒就自动保存在.swp文件中。
通常的恢复步骤是,移动到文件所在目录下;使用vim –r [filename] 恢复filename文件,如果编辑的文件没有保存使用 vim –r “” 即可;把恢复的文件另存为另一个文件,使用diff程序(可以使用vimdiff)查看是否正确恢复。如果不记得文件名,使用 vim –r 列出所有交换文件。
3. 开发常用命令
3.1. tag
有tag后能够非常方便的在多个源代码文件中进行跳转,这个特性非常方便浏览、阅读代码。
C/C++源程序可以使用 ctags -R *.c *.cpp 产生tag文件,ctag已经包含在最新的vim发布版本中。有了tag文件,就可以方便的跳转了。
使用vim –t [tagname] 启动vim可以直接打开相应的文件,并定位到tagname的位置。在vim里使用命令 :tag [tagname] 可以跳转到tagname的定义,tagname可以使用正则表达式来进行模糊匹配。
使用ctrl-] 可以直接跳动到当前词的定义,如果当前词不是tag,使用右边的第一个tagname。这个跳动过程组成tag堆栈,沿堆栈跳回的命令是 ctrl-t ,沿堆栈上下移动的命令是 :tn (老的项),:tp (新的项)。列出tag堆栈使用 :tags 。
3.2. 预览窗口
调用函数时,通常非常需要查看函数的参数,使用tag技术和预览窗口可以方便的实现这个功能。
命令 :ptag [tagname] 在预览窗口显示跳转到tagname定义。当前词作为tagname,可以直接输入 ctrl-w } ,在预览窗口查看当前词的定义。
关闭预览窗口使用 :pclose 命令。
在预览窗口打开一个文件,使用 :pedit [filename] 命令。在当前文件和包含文件中搜索某个词,并显示到预览窗口中,可以使用 :psearch [name]。通常使用这个命令搜索函数原型。
3.2. 移动
vim缺省非常好的支持多种缩进方式。调整缩进可以使用 < (向左缩进)或者 >(向右缩进),前面可以加范围,或者后面加motion。
使用 ={motion} 命令对一块文字进行自动缩进。
找到对应的{}、[]、(),有时候比较麻烦,使用 % 命令可以非常方便找到对应的符号,除了上面的几组符号,还可以是 /* */ 或者 #if #ifdef #else #elif #endif 等。
代码块通常是由{}包括的,在各个代码块之间移动是非常方便的。[{ 移动到光标所在块首,]} 移动到所在块尾。使用命令 [[ 跳转到最外层的{,]] 跳转到最外层的},最外层通常是C/C++程序的函数边界。第二层的{使用 [m 命令,第二层的}使用 ]m 命令,通常java程序第二层才是函数的边界。
在文件内或者文件间跳转,有时候需要返回上次光标所在位置。可以使用ctrl-o 命令跳转到以前的位置,ctrl-i 命令跳转到新的位置。使用命令 :jumps 列出所有跳转列表。这些跳转命令包括跳转到标记、跳转到tag、搜索、替换、%、多于一行或一列的移动等。
3.3. 帮助
编写程序时,经常需要使用 man [section] funcname 来查看某个系统函数的帮助,在vim中使用 [section] K 命令直接查看当前词的manpage。
如果确实需要执行shell命令,可以在不离开vim环境和当前终端的情况下,使用 :!<cmd> 来执行一个shell命令,要获得一个shell,可以使用 :shell 。
3.4. 编译
说vim可以是集成开发环境,就是它可以方便的实现编辑-编译-编辑的循环。如果你的程序有makefile,使用 :make 命令来进行编译,而且更方便的是vim可以识别编译输出的错误,并定位到相应的位置。命令 :cc [num] 显示第num个完整的错误消息。
在错误列表中移动,可以使用 :cn (下一个错误)和 :cp (上一个错误)。列出整个错误列表使用命令 :clist 。
Vim保存多个错误列表,老的错误列表使用 :colder ,新的错误列表使用 :cnewer 。
这些命令被称为QuickFix命令,上面介绍的grep命令也使用这些命令。