第一次听说这个插件还是在偶然的情况下看到别人的博客,听说了这个插件的大名.本来打算在实训期间来完成安装的,无奈网实在不给力,也就拖到了回家的时候.在开始准备工作的时候就了解到这个插件不是很容易安装,安装的时候果然名不虚传.(关于这方面的内容,请查看另一篇文章)不过,有付出总有回报,安装之后用上这个插件,真心为这个插件的强大所折服.
那这个插件有何不同?
YouCompleteMe的特别之处
基于语义补全
总所周知,Vim
是一款文本编辑器.也就是说,其最基础的工作就是编辑文本,而不管该文本的内容是什么.在Vim
被程序员所使用后,其慢慢的被肩负了与IDE一样的工作,文本自动补全(ie.acp
,omnicppcompleter
),代码检查(Syntastic
)等等工作.
针对文本自动补全这个功能来说,主要有两种实现方式.
- 基于文本
我们常用的omnicppcompleter
,acp
,vim自带的c-x, c-n
的实现方式就是基于文本.更通俗的说法,其实就是一个字:
猜
其通过文本进行一些正则表达式的匹配,再根据生成的tags(利用ctags
生成)来实现自动补全的效果.
- 基于语义
顾名思义,其是通过分析源文件,经过语法分析以后进行补全.由于对源文件进行分析,基于语义的补全可以做到很精确.但是这显然是vim所不可能支持的.而且经过这么多年发展,由于语法分析有很高的难度,也一直没有合适的工具出现.直到,由apple支持的clang/llvm
横空出世.YouCompleteMe
也正是在clang/llvm
的基础上进行构建的.
整合实现了多种插件
- clang_complete
- AutoComplPop
- Supertab
- neocomplcache
- Syntastic(类似功能,仅仅针对c/c++/obj-c代码)
支持语言
- c
- c++
- obj-c
- c#
- python
对于其他的语言,会调用vim设置的omnifunc
来匹配,因此同样支持php
,ruby
等语言.
已知的有 * javascript —-tern_for_vim * ruby/java —-eclim
使用效果图
使用感受
- 和IDE一样,自动补全,
- 根据
include
的文件进行补全 - 不用再蹩脚的生成tags
- 补全非常精准,而且速度很快,不会有延迟(以前在大项目上,acp用起来实在是很卡)
- 支持类似tags的跳转,跳到定义处以及使用处
- 出错提示很智能,并且用起来真的是如丝般柔滑,不用输入
:w
进行强制检测
安装
说完了那么多好处,就要说到安装了.不同于以往其他vim插件,YCM是一款编译型的插件.在下载完后,需要手动编译后才能使用.对应其他的插件来说,仅仅就是把.vim的文件丢到相应文件夹下就可以.而这也加大了使用YCM的难度.
安装准备
- 最新版的Vim(7.3.584+),编译时添加+python标志(已经安装的可以通过
vim --version
查看) - cmake(ubuntu可以通过
sudo
)
apt-get install build-essential cmake来安装 - python头文件(ubuntu可以通过sudo apt-get install python-dev python3-dev来安装)
- 安装vundle插件,用于安装管理vim的插件
按照作者在https://github.com/gmarik/Vundle.vim#about提供的方法安装好vundle。主要就是下载
ubuntu下自动安装
在.vimrc
中的
call vundle#begin() 与call vundle#end() 之间
添加下列代码
Plugin ‘Valloric/YouCompleteMe‘
保存退出后打开vim,在正常模式下输入
:PluginInstall
等待:PluginInstall将YouCompleteMe安装完成
而后进行编译安装:
cd ~/.vim/bundle/YouCompleteMe
./install --clang-completer
如果不需要c-family的补全,可以去掉--clang-completer
.如果需要c#
的补全,请加上--omnisharp-completer
.
正常来说,YCM会去下载clang的包,如果已经有,也可以用系统--system-libclang
.
就这样,安装结束.打开vim,输入如下命令:
1 | :YcmDiags |
如果没有提示错误,那就说明插件安装好了。
如果没有提示YCM未编译,则说明安装已经成功了.ubuntu下手动编译安装
安装的脚本并不是什么时候都好用,至少对我来说是这样的.安装完之后出现了问题,参考issue#809.
在用
:PluginInstall安装完成后使用
git clone --recursive https://github.com/Valloric/YouCompleteMe.git
获取最新的仓库,而后使用git submodule update --init --recursive
确认仓库的完整性后,开始安装流程.
- 安装cmake(ubuntu下使用sudo apt-get install build-essential cmake)
- 安装python-dev.(ubuntu下使用
sudo
)
apt-get install python-dev python3-dev -
编译
./install --clang-completer
这样就完成了,开始感受YCM提供的完全不逊色于大型IDE所提供的自动补全功能吧.
配置
不同于很多vim插件,YCM首先需要编译,另外还需要有配置.在vim启动后,YCM会找寻当前路径以及上层路径的.ycm_extra_conf.py
.在~/.vim/bundle/YouCompleteMe/cpp/ycm/.ycm_extra_conf.py
中提供了默认的模板.也可以参考我的(就在模板上改改而已).不过这个解决了标准库提示找不到的问题.
一般来说,我会在~
目录下放一个默认的模板,而后再根据不同的项目在当前目录下再拷贝个.ycm_extra_conf.py
.
主要就是设置-I与-isystem的选项
import
os
import
ycm_core
flags
=
[
'-std=c++11'
,
'-stdlib=libc++'
,
'-Wno-deprecated-declarations'
,
'-Wno-disabled-macro-expansion'
,
'-Wno-float-equal'
,
'-Wno-c++98-compat'
,
'-Wno-c++98-compat-pedantic'
,
'-Wno-global-constructors'
,
'-Wno-exit-time-destructors'
,
'-Wno-missing-prototypes'
,
'-Wno-padded'
,
'-x'
,
'c++'
,
'-I'
,
'.'
,
'-isystem'
,
'D:/android-ndk-r9c/platforms/android-19/arch-arm/usr/include/sys'
,
'-isystem'
,
'D:/android-ndk-r9c/platforms/android-19/arch-arm/usr/include'
,
'-isystem'
,
'D:/android-ndk-r9c/sources/cxx-stl/llvm-libc++/libcxx/include'
,
'-I'
,
'D:/work/jni/inc'
,
'-I'
,
'D:/work/jni/common'
,
]
compilation_database_folder
=
''
if
compilation_database_folder:
database
=
ycm_core.CompilationDatabase( compilation_database_folder )
else
:
database
=
None
SOURCE_EXTENSIONS
=
[
'.cpp'
,
'.cxx'
,
'.cc'
,
'.c'
,
'.m'
,
'.mm'
]
def
DirectoryOfThisScript():
return
os.path.dirname( os.path.abspath( __file__ ) )
def
MakeRelativePathsInFlagsAbsolute( flags, working_directory ):
if
not
working_directory:
return
list
( flags )
new_flags
=
[]
make_next_absolute
=
False
path_flags
=
[
'-isystem'
,
'-I'
,
'-iquote'
,
'--sysroot='
]
for
flag
in
flags:
new_flag
=
flag
if
make_next_absolute:
make_next_absolute
=
False
if
not
flag.startswith(
'/'
):
new_flag
=
os.path.join( working_directory, flag )
for
path_flag
in
path_flags:
if
flag
=
=
path_flag:
make_next_absolute
=
True
break
if
flag.startswith( path_flag ):
path
=
flag[
len
( path_flag ): ]
new_flag
=
path_flag
+
os.path.join( working_directory, path )
break
if
new_flag:
new_flags.append( new_flag )
return
new_flags
def
IsHeaderFile( filename ):
extension
=
os.path.splitext( filename )[
1
]
return
extension
in
[
'.h'
,
'.hxx'
,
'.hpp'
,
'.hh'
]
def
GetCompilationInfoForFile( filename ):
if
IsHeaderFile( filename ):
basename
=
os.path.splitext( filename )[
0
]
for
extension
in
SOURCE_EXTENSIONS:
replacement_file
=
basename
+
extension
if
os.path.exists( replacement_file ):
compilation_info
=
database.GetCompilationInfoForFile( replacement_file )
if
compilation_info.compiler_flags_:
return
compilation_info
return
None
return
database.GetCompilationInfoForFile( filename )
def
FlagsForFile( filename,
*
*
kwargs ):
if
database:
compilation_info
=
GetCompilationInfoForFile( filename )
if
not
compilation_info:
return
None
final_flags
=
MakeRelativePathsInFlagsAbsolute(
compilation_info.compiler_flags_,
compilation_info.compiler_working_dir_ )
else
:
relative_to
=
DirectoryOfThisScript()
final_flags
=
MakeRelativePathsInFlagsAbsolute( flags, relative_to )
return
{
'flags'
: final_flags,
'do_cache'
:
True
}
YouCompleteMe提供的其他功能
YCM除了提供了基本的补全功能,自动提示错误的功能外,还提供了类似tags的功能:
- 跳转到定义
GoToDefinition
- 跳转到声明
GoToDeclaration
- 以及两者的合体
GoToDefinitionElseDeclaration
可以在.vimrc中配置相应的快捷键.
nnoremap <leader>gl :YcmCompleter GoToDeclaration<CR>
nnoremap <leader>gf :YcmCompleter GoToDefinition<CR>
nnoremap <leader>gg :YcmCompleter GoToDefinitionElseDeclaration<CR>
另外,YCM也提供了丰富的配置选项,同样在.vimrc中配置.具体请参考
let g:ycm_error_symbol = '>>'
let g:ycm_warning_symbol = '>*'
同时,YCM可以打开location-list来显示警告和错误的信息:YcmDiags
.个人关于ycm的配置如下:
" for ycm
let g:ycm_error_symbol = '>>'
let g:ycm_warning_symbol = '>*'
nnoremap <leader>gl :YcmCompleter GoToDeclaration<CR>
nnoremap <leader>gf :YcmCompleter GoToDefinition<CR>
nnoremap <leader>gg :YcmCompleter GoToDefinitionElseDeclaration<CR>
nmap <F4> :YcmDiags<CR>
" YouCompleteMe 功能
" 补全功能在注释中同样有效
let g:ycm_complete_in_comments
=
1
" 允许 vim 加载 .ycm_extra_conf.py 文件,不再提示
let g:ycm_confirm_extra_conf
=
0
" 开启 YCM 基于标签引擎
let g:ycm_collect_identifiers_from_tags_files
=
1
" 引入 C
+
+
标准库tags,这个没有也没关系,只要.ycm_extra_conf.py文件中指定了正确的标准库路径
set
tags
+
=
/
data
/
misc
/
software
/
misc.
/
vim
/
stdcpp.tags
" YCM 集成 OmniCppComplete 补全引擎,设置其快捷键
inoremap <leader>; <C
-
x><C
-
o>
" 补全内容不以分割子窗口形式出现,只显示补全列表
set
completeopt
-
=
preview
" 从第一个键入字符就开始罗列匹配项
let g:ycm_min_num_of_chars_for_completion
=
1
" 禁止缓存匹配项,每次都重新生成匹配项
let g:ycm_cache_omnifunc
=
0
" 语法关键字补全
let g:ycm_seed_identifiers_with_syntax
=
1
" 修改对C函数的补全快捷键,默认是CTRL
+
space,修改为ALT
+
;
let g:ycm_key_invoke_completion
=
'<M-;>'
" 设置转到定义处的快捷键为ALT
+
G,这个功能非常赞
nmap <M
-
g> :YcmCompleter GoToDefinitionElseDeclaration <C
-
R>
=
expand(
"<cword>"
)<CR><CR>
YCM提供的跳跃功能采用了vim的jumplist
,往前跳和往后跳的快捷键为Ctrl+O
以及Ctrl+I
.
总结
YouCompleteMe
是我用过的最爽的一个自动补全的插件了.之前使用acp时,遇到大文件基本上就卡死了,以至于都不怎么敢使用.由于YCM使用的时C/S结构,部分使用vim脚本编写,部分认为原生代码,使得跑起来速度飞快.
抛弃Vim自带的坑爹的补全吧,抛弃ctags
吧,抛弃cscope
吧,YCM才是终极的补全神器
.
在安装过程中,我也遇到了不少的坑.一会会发一篇解决这些坑的文章.
最后祝大家码年顺利,一码平川,码到功成.
一些用户对于ycm的讨论: