13

<< 返回vim使用进阶: 目录

本节所用命令的帮助入口:

:help quickfix
:help :make
:help 'makeprg'
:help 'errorformat'
:help 'switchbuf'
:help location-list
:help grep
:help :vimgrep
:help :grep
:help starstar-wildcard 

以前读武侠小说,看到武林高手们都是从来剑不离手的。使用vim写程序,你也可以做到这一点,:-)

vim由一个程序员开发,而且为更多的程序员所使用,所以在vim中加强了对软件开发的支持,quickfix模式的引入就是一个例子。所谓quickfix模式,它和Normal模式、Insert模式没什么关系,它只是一种加快你开发速度的工作方式。

Quickfix模式的主要思想是保存一个位置列表,然后提供一系列命令,实现在这个位置列表中跳转。

位置列表的产生可以从编译器的编译输出信息中获得,也可以由grep命令的输出信息中获得,我们上篇文章所介绍的cscope命令,也可以产生位置列表信息(:help ‘cscopequickfix’)。

[编译]

通常,我们在开发过程中,经常要写代码,编译,修改编译错误,这个过程会数十遍上百遍的重复。如果你是根据编译器输出的错误信息,打开出错的文件,找到出错的行,然后再开始修改,那效率未免太低下了。

利用vim的quickfix模式,可以大大加快这一过程,你可以在vim启动编译,然后vim会根据编译器输出的错误信息,自动跳到第一个出错的地方,让你进行修改;修改完后,使用一个快捷键,跳到下一个错误处,再进行修改,方便的很。

为了做到这一点,你首先要定义编译时所使用的程序,对大多数使用Makefile的项目来说,vim的缺省设置”make“已经可以满足要求了。如果你的项目需要用一个特殊的程序进行编译,就需要修改’makeprg‘选项的值。

大家在学编程时大概都读过”hello world”程序,我们就以这个简单的例子为例,讲一下quickfix模式的用法。

该程序的内容如下,里面包含了三个小小的错误:

/* hello world demo */
#include <stdio.h"
int main(int argc, char **argv)
{
    int i;
    print("hello world\n");
    return 0;
} 

我们可以为这个程序写个小小的Makefile文件,不过为了演示’makeprg‘的设置方法,我们并不用Makefile,而直接设置’makeprg‘选项,如下:

:set makeprg=gcc\ -Wall\ -ohello\ hello.c 

上面的命令会把hello.c编译为名hello的可执行文件,并打开了所有的Warnning。如果编译命令中有空格,需要使用’\‘对空格进行转义,上面的例子就使用了’\‘转义空格。

我们设置好’makeprg‘选项后,输入下面的命令就可以编译了:

:make 

在使用”:make“时,vim会自动调用’makeprg‘选项定义的命令进行编译,并把编译输出重定向到一个临时文件中,当编译出现错误时,vim会从上述临时文件中读出错误信息,根据这些信息形成quickfix列表,并跳转到第一个错误出现的地方。

对于我们上面的程序来说,光标会停在第三行,也就是第一个出错的位置,vim同时会提示出错信息。如果你没看清出错信息,可以输入”:cc“命令,vim会更次显示此信息,或者干脆使用”:cw“命令,打开一个quickfix窗口,把所有的出错信息显示出来,见下图:

现在我们知道错在哪儿了,修正一下,然后使用”:cn“命令(或者在Quickfix List对应行上输入回车)跳到下一个出错的地方,以此类推,直到修正全部错误。

好了,千辛万苦,我们的hello world终于工作了。乍一看这个例子,似乎Quickfix并没有提高什么效率,但如果你的错误出现在多个不同目录的不同文件里,它可以帮你省很多时间,使你可以集中精力在修正bug上。

vim可以同时记住最新的10个错误列表,也就是说你最近10次使用”:make“命令编译所遇到的错误都保存着,可以使用”:colder“和”:cnewer“命令,回到旧的错误列表,或者到更新的错误列表。

在quickfix模式里经常用到的命令有:

:cc                显示详细错误信息 ( :help :cc )
:cp                跳到上一个错误 ( :help :cp )
:cn                跳到下一个错误 ( :help :cn )
:cl                列出所有错误 ( :help :cl )
:cw                如果有错误列表,则打开quickfix窗口 ( :help :cw )
:col               到前一个旧的错误列表 ( :help :col )
:cnew              到后一个较新的错误列表 ( :help :cnew ) 

更多的命令,以及这些命令更详细的解释,请参见手册。

对于经常用到的命令,最好提供更方便的使用方法,在我的vimrc中的定义:

autocmd FileType c,cpp  map <buffer> <leader><space> :w<cr>:make<cr>
nmap <leader>cn :cn<cr>
nmap <leader>cp :cp<cr>
nmap <leader>cw :cw 10<cr> 

现在使用”,<space>“(先按,再按空格)就可以编译,使用”,cp“和”,cn“跳到上一个和下一个错误,使用”,cw“来打开一个quickfix窗口。这下顺手多了!

如果你希望跳转到出错的文件时,使用一个分隔的窗口打开,请参阅’switchbuf‘选项的值。

在vim7中,每个窗口都可以拥有自己的位置列表,这样,你就能够同时打开多个位置列表了,而quickfix列表在整个vim中只有一个。你可以使用位置列表来显示编译错误信息,具体命令参阅手册:”:help location-list“以及”:help :lmake“。

[GREP]

我们在程序员的利器 – cscope中讲过,cscope可以做为一个快速的grep程序使用,对于我们的软件项目,用cscope生成一个数据库,可以大大加快查找字符串的速度。但cscope需要事先生成一个数据库,对一些简单的查找,不需要专门为之生成数据库,这时候可以使用grep。

Grep的名字来源于”g/re/p”,”re”是正则表达式(regex)的意思,”p”是打印,也就是把匹配正则表达式的行打印出来。

vim既可以使用外部的grep程序,也可以使用内部集成的grep功能。

使用集成的grep命令非常简单,通常使用格式为:

:vimgrep /main/gj **/*.c 

在上面的例子里,我们使用vim内部集成的grep功能,在当前目录及其子目录树的所有c文件中查找main字符串,如果一行中main出现了多次,每个匹配都计入;在查找到后,不立即跳转到第一个匹配的地方。

使用内部集成的grep功能速度要比外部grep慢一些,因为它会打开每个文件,对其进行检查,然后关闭;但集成的grep支持vim增强的正则表达式,可以利用它进行更为复杂的查找。它也支持vim扩展的文件通配符表示方式,见”:help starstar-wildcard“。

vimgrep查找到的结果,也会放在quickfix列表中。下图是在vim 7.0的源代码目录中执行上面的命令生成的quickfix列表:

我们可以使用上面介绍的quickfix模式的命令,来查看这些匹配。

你也可以用外部的grep程序来查找,如果你的系统中所用的不是标准的grep程序,那么就需要修改’grepprg‘选项,详情请参阅手册。

使用外部grep的语法与grep程序相同,请参阅grep的手册。

无论使用内部的vimgrep,还是使用外部的grep,vim都允许你将查找到的结果放在与窗口相关联的位置列表,要了解详细信息,”:help :lvimgrep“及”:help :lgrep“。

在我的vimrc中,定义下面的键映射,利用它可以在当前文件中快速查找光标下的单词:

nmap <leader>lv :lv /<c-r>=expand("<cword>")<cr>/ %<cr>:lw<cr> 

[参考文档]

<< 返回vim使用进阶: 目录

原创文章,请阅读页脚的许可方式,转载请注明:转载自易水博客 [ http://easwy.com/blog/ ]

本文链接地址: http://easwy.com/blog/archives/advanced-vim-skills-quickfix-mode/

文章的脚注信息由WordPress的wp-posturl插件自动生成

文本处理

9条评论 添加评论

  1. Easwy
    2009/02/13 15:46 | #1

    (以下为CSDN评论的转帖)

    #easwy 发表于2007-06-01 09:58:51 IP: 213.70.90.*
    首先定义你的mapleader是什么,例如:
    let mapleader = “,”

    然后,对上面的映射,输入”,lv”,就会查找当前光标下的单词,然后在窗口的位置列表里显示出来。

    :help mapleader

    #xiaoqiang 发表于2008-06-01 15:24:07 IP: 125.46.31.*
    最后设的那个快捷键我使了一下,可是只是在一个location列表中显示本文件中搜索结果。。我想查找当前光标下的单词,显示本文件夹中所有文件的搜索结果应该怎么设置呢?
    谢谢。。。

    2008-06-11 14:17:56作者回复
    nmap <silent> <leader>ld :lv /<c-r>=expand(“<cword>”)<cr>/ *<cr>:lw<cr>

    #xiaoqiang 发表于2008-06-07 13:37:32 IP: 162.105.75.*
    我搞定了。。。thanks all the same

  2. Easwy
    2009/02/13 15:47 | #2

    (以下为CSDN评论的转帖)

    #新人 发表于2008-07-04 20:27:03 IP: 162.105.75.*
    请问您用的配色方案是什么?就是这个墨绿色背景
    的?十分喜欢,谢谢。。

    #easwy 发表于2008-07-07 16:06:27 IP: 213.70.90.*
    参见我的另外一篇文章:
    http://blog.csdn.net/easwy/archive/2007/03/29/1545707.aspx

    #Canbus 发表于2008-08-08 16:11:16 IP: 202.101.116.*
    nmap <silent> <leader>ld :lv /<c-r>=expand(“<cword>”)<cr>/ *<cr>:lw<cr>
    ——–
    这个一堆的乱码,能否再贴一次,3q
    ==============
    另这个快截键怎么用呢?
    nmap lv :lv /=expand(““)/ %:lw

    是不是在command方式下输入lv呢?但是我输入完后显示E471:argument required

    菜菜鸟一个,见笑了

    2008-10-29 15:36:32作者回复
    在normal模式下用,就是按ESC后的模式,输入”,lv”

  3. 2009/03/24 12:41 | #3

    Hi..大哥
    找到一个我很需要的错误检查形式。。但是我按照他的步骤却没办法弄出来
    http://www.koch.ro/blog/index.php?url=archives/62-Integrate-PHP-CodeSniffer-in-VIM.html&serendipitycsuccess=moderate#feedback
    能否抽时间帮我看看。。。
    我已经将函数放到.vimrc文件里了。。。也不知道哪出问题了。。反正就是没反应

  4. 2009/03/24 18:32 | #4

    @SamPeng
    试试 :Phpcs

  5. xyf
    2009/04/11 21:00 | #5

    大哥,你的命令:
    nmap lv :lv /=expand(“”)/ %:lw
    可以查找当前文件查找光标下的单词,如何在某个目录里递归并在指定的某些文件类型(比如.c .s等)里查找某个单词?
    在公司里是使用windows,grep.vim的插件在windows下Grep功能正常,Rgrep的功能不正常。在cygwin里的vim中倒是可以正常使用,不过cygwin中的vim速度慢了些,没有win版的gvim来得快(而且console模式下colorscheme不好看^_^),google了好久,并没有相关的解决方案,grep.vim的作者好像也消失了。
    不知道您平时是如何做到相关功能的,非常感谢!!!

  6. 2009/04/11 23:39 | #6

    @xyf
    vim自已就带有grep功能,:help vimgrep
    我前面有一篇文章也有介绍vimgrep:vim使用进阶: 剑不离手 – quickfix

  7. 2009/04/23 22:53 | #7

    [...] vim使用进阶: 乱花渐欲迷人眼 – 语法高亮 vim使用进阶: 程序员的利器 – cscope vim使用进阶: 剑不离手 – quickfix vim使用进阶: 智能补全 vim使用进阶: 自动补全 vim使用进阶: [...]

  8. squall
    2009/11/11 19:19 | #8

    请问怎么让set makeprg 和make在一个函数中全部实现?
    func! QuickfixGcc()
    exec “w”
    exec “!set makeprg=gcc\ -Wall\ -o%<\ %”
    exec “!make”
    exec “cw”
    endfunc

    请文这样怎么错了阿 谢谢~~

  9. 2009/11/12 18:55 | #9

    @squall
    !set是执行一个外部的set的命令
    !make也是在执行外部的make命令

    把这两条命令前的!去掉
    :help :!

评论本文