美文网首页
为YCM配置ycm_extra_conf.py脚本

为YCM配置ycm_extra_conf.py脚本

作者: 哈莉_奎茵 | 来源:发表于2018-06-03 17:50 被阅读0次

    前言

    YCM应该是vim补全的标配了,关于配置可以直接阅读官方README文件,写得十分详细,但是需要点耐心。
    我之前写过Ubuntu 16.04 64位安装YouCompleteMe,主要针对的是apt-get安装的vim版本不支持YCM的python引擎问题,所以得手动编译vim源码。不过关于YCM的配置文件很多都是直接照抄的,有些用法没有理解,最近想要看看开源代码,由于include的头文件很多都把项目目录作为包含目录,而YCM配置文件缺只使用了系统目录,所以无法进行代码跳转和补全。

    修改YCM配置文件

    YCM配置文件的路径在.vimrc文件中定义,比如我的配置是
    let g:ycm_global_ycm_extra_conf='~/.ycm_extra_conf.py'
    用的默认命名ycm_extra_conf.py,直接扔到了用户主目录下,修改文件名和路径都是可以的。该配置文件的核心就是修改flags变量。

    flags = [
    '-Wall',
    '-Wextra',
    '-Werror',
    '-Wno-long-long',
    '-Wno-variadic-macros',
    '-fexceptions',
    '-DNDEBUG',
    '-std=c++11',
    '-x',
    'c++',
    '-I',
    '/usr/include',
    '-isystem',
    '/usr/lib/gcc/x86_64-linux-gnu/5/include',
    '-isystem',
    '/usr/include/x86_64-linux-gnu',
    '-isystem'
    '/usr/include/c++/5',
    '-isystem',
    '/usr/include/c++/5/bits'
    ]
    

    可以看到flags是一个列表,参数是和gcc的参数一致。
    比如对下面的示例项目

    $ tree sample/
    sample/
    ├── include
    │   └── foo.h
    └── src
        └── main.cc
    

    其中main.cc的include代码如下


    可以发现vim提示foo.h无法找到,但是我们可以这样编译该文件
    g++ main.cc -isystem $HOME/sample/include
    也可以用下列编译方式
    g++ main.cc -I $HOME/sample/include

    man gcc查看手册,找到-isystem的说明

           -isystem dir
               Search dir for header files, after all directories specified by -I
               but before the standard system directories. 
    

    搜索头文件的顺序是:-I指定目录、-isystem指定目录、标准系统目录。
    OK,回顾之前flags参数的设置,一目了然,只需要在末尾添加新的-I-isystem,后接头文件目录路径即可。在python中可以定义一个变量保存用户主目录路径
    home_dir = os.environ['HOME']
    需要注意的细节是,flags列表最后一项的的结尾没有加逗号,如果在后面添加新项,记得加上逗号(我好几次忘记加了导致语法错误)

    # 错误的代码
    '/usr/include/c++/5/bits'  # 粗心大意,直接在后面换行,忘记加逗号。
    '-isystem',
    os.path.join(home_dir, 'sample/include')
    ]
    

    拷贝YCM配置文件到项目目录

    之前修改的是全局YCM配置文件,如果有多个项目,每次都修改全局配置文件的话,flags列表就会越来越长,头文件搜索目录越来越多,会影响YCM的效率,因此可以拷贝一份到项目目录下,然后再进行修改。

    ~/sample$ cp ~/.ycm_extra_conf.py .
    ~/sample$ ls -a
    .  ..  include  src  .ycm_extra_conf.py
    

    YCM配置文件的查找顺序是当前目录>上层目录>...>根目录>YCM全局目录。
    然后问题来了,每次打开文件都会提示是否载入YCM配置文件。


    这问题十分烦人,不过也给出了提示,想要关闭这个提示可以参考YCM文档。找到的解决方案如下
    let g:ycm_confirm_extra_conf = 0
    官方说明表示该选项的默认值为1是为了阻止不是由你写的YCM配置代码,比如你从网上下载一个项目,由于YCM配置文件默认是隐藏文件,所以你也不知道项目目录下包含了YCM配置文件,此时vim打开代码时就会提示你是否载入,从而提醒你项目目录里有YCM配置文件。
    所以安全的做法是自己写代码时,可以把该选项设为0,其余情况下保持默认值1。

    实践:阅读Linux内核源码的配置

    这几天想确认下内核是否真的用do_fork来区分线程和进程,于是下载了源码来查找。首先找到kernel/fork.c,vim打开后如下所示


    借助find命令来查找头文件的位置
    ~/linux-4.16.13$ find . -name "mm.h"
    ./drivers/gpu/drm/nouveau/include/nvkm/core/mm.h
    ./include/linux/decompress/mm.h
    ./include/linux/mm.h
    ./include/linux/sched/mm.h
    ./arch/unicore32/mm/mm.h
    ./arch/arm/mm/mm.h
    ./tools/testing/scatterlist/linux/mm.h
    ./tools/include/linux/sched/mm.h
    

    find默认就是递归查找,可以用maxdepth选项来控制递归最大深度,比如

    ~/linux-4.16.13$ find . -maxdepth 3 -name "mm.h"
    ./include/linux/mm.h
    

    于是可以把YCM配置文件拷贝到linux-4.16.13目录下,修改flags如下

    flags = [
    # 省略之前的参数
    '-isystem',
    os.path.join(home_dir, 'linux-4.16.13'),
    '-isystem',
    os.path.join(home_dir, 'linux-4.16.13/include')
    ]
    

    这里还不够,只是解决了头文件包含的问题,代码跳转时,比如光标停在task_struct,点击跳转快捷键时,发现无法跳转。这里就需要用grep命令来查找,

    ~/linux-4.16.13/include$ egrep -rn "struct +task_struct +{"
    linux/sched.h:524:struct task_struct {
    

    egrep即使用扩展正则表达式,内核编码风格是把左大括号和结构体名称放在一行,所以用这个正则表达式就找到了task_struct的定义。
    可以用sed命令多看几行来确认下

    ~/linux-4.16.13/include$ sed -n '524,530p' linux/sched.h 
    struct task_struct {
    #ifdef CONFIG_THREAD_INFO_IN_TASK
        /*
         * For reasons of header soup (see current_thread_info()), this
         * must be the first element of task_struct.
         */
        struct thread_info      thread_info;
    

    嗯,定义了thread_info成员,继续把该目录加入flags中。但还是跳转不了,因为发现fork.c中并未包含linux/sched.h。这种情况暂时也没找到好的方法,只能用grep命令来手动查找。

    .vimrc中的其他YCM配置选项

    之前的配置文件是东抄抄西抄抄凑的,这次整理了下,给每个选项设置都加了注释。

    " YouCompleteMe
    " Python Semantic Completion
    let g:ycm_python_binary_path = '/usr/bin/python3'
    " C family Completion Path
    let g:ycm_global_ycm_extra_conf='~/.ycm_extra_conf.py'
    " 跳转快捷键
    nnoremap <c-k> :YcmCompleter GoToDeclaration<CR>|
    nnoremap <c-h> :YcmCompleter GoToDefinition<CR>| 
    nnoremap <c-j> :YcmCompleter GoToDefinitionElseDeclaration<CR>|
    " 停止提示是否载入本地ycm_extra_conf文件
    let g:ycm_confirm_extra_conf = 0
    " 语法关键字补全
    let g:ycm_seed_identifiers_with_syntax = 1
    " 开启 YCM 基于标签引擎
    let g:ycm_collect_identifiers_from_tags_files = 1
    " 从第2个键入字符就开始罗列匹配项
    let g:ycm_min_num_of_chars_for_completion=2
    " 在注释输入中也能补全
    let g:ycm_complete_in_comments = 1
    " 在字符串输入中也能补全
    let g:ycm_complete_in_strings = 1
    " 注释和字符串中的文字也会被收入补全
    let g:ycm_collect_identifiers_from_comments_and_strings = 1
    " 弹出列表时选择第1项的快捷键(默认为<TAB>和<Down>)
    let g:ycm_key_list_select_completion = ['<C-n>', '<Down>']
    " 弹出列表时选择前1项的快捷键(默认为<S-TAB>和<UP>)
    let g:ycm_key_list_previous_completion = ['<C-p>', '<Up>']
    " 主动补全, 默认为<C-Space>
    "let g:ycm_key_invoke_completion = ['<C-Space>']
    " 停止显示补全列表(防止列表影响视野), 可以按<C-Space>重新弹出
    "let g:ycm_key_list_stop_completion = ['<C-y>']
    

    相关文章

      网友评论

          本文标题:为YCM配置ycm_extra_conf.py脚本

          本文链接:https://www.haomeiwen.com/subject/vipjsftx.html