背景

  由于不可抗拒的原因,学习环境由之前centos的一台机器上,变成了ubuntu的一台机器上。因此,需要在新的ubuntu的机器上再配置一次vim环境。算起来这已经是第三次配置vim环境了(mac上配过一次,centos上配过一次,ubuntu上再配置一次)。这次配置大体上比较顺利,还是沿用之前日志用的方法(http://www.cnblogs.com/xbf9xbf/p/4860484.html),用pathogen管理插件;而且这次ubuntu机器能联网,比之前在centos上无网的情况下方便很多。

  其中,稍有难度的插件就是YouCompleteMe(YCM)插件。之前趟过YCM的坑了(http://www.cnblogs.com/xbf9xbf/p/4876306.html),这次又遇到了些新的问题。

问题

  YCM需要在本机编辑才能够install,这个过程就参照之前的完成的。(下载ubuntu对应版本的clang+llvm的预编译文件,编译YCM,install的时候注意要加入--clang-completer来支持C-family的smantic)。但是,在搞一个.c文件的时候遇到了点儿YCM的补全的问题。

  简单说,问题就是:用vim编辑.c文件的时候,ycm能够补全user define的结构体,但是却无法补全<netdb.h>中的struct addrinfo系统结构体。

  能够补全user defined的结构体:

  

  不能补全<netdb.h>中的struct addrinfo结构体:

  

Solution:

  这个问题现在已经解决了,记录下当时解决问题的一些逻辑轨迹,从中吸取经验。

  思路一:ycm的编译过程出了问题。

      ycm编译过程真的没错:支撑的现象有(1)ycm编译过程中没有报错 (2)vim中ycm可以补全user define的结构体。

      ycm编译过程真的出错:猜想的原因(1)可能是各种clang+llvm版本的问题,后面用3.7重新编译了也米有问题(2)ubuntu系统的问题

      如果真的没错,那么就不用考虑编译过程;如果真的出错了,这种跟编译器版本和系统implementation的问题也是我力所不及的。

      因此,这个思路的结论是,必须PASS这个方面solution。

  思路二:ycm的配置出了问题。

      以为是.ycm_extra_conf.py的配置出了问题:主要集中在flags选项中的'-isystem',是不是该include的没有include进去呢?

      这个查了好久,并没有什么实质的斩获。当时是PASS了。

  思路三:ubuntu系统的netdb.h文件的问题

      同样的代码,在我的mac和原来的centos上就可以补全出来,但是在ubuntu上就不能补全出来。一个主要变数就是netdb.h文件。

      可能是netdb.h中没有这个struct addrinfo呢?vi /usr/include/netdb.h查看一下,确实有这个结构体。PASS。

  思路四:文件访问权限的问题

      可能是我的代码没有访问netdb.h的权限。sudo chmod 777之后,还是不行,看来不是权限的问题。

      直接把/usr/include/netdb.h文件放在工程文件的同一个目录下,.c文件直接#include "netdb.h"。

      这样不存在文件和文件夹的权限问题了,结果还是悲剧,无法补全。

  上面四个思路都否定之后,有些束手无策了。

  最后放了一个大招,直接把mac上的/usr/include/netdb.h直接scp到ubuntu的工程文件同个目录下,.c文件还是#include "netdb.h"

  这样做之后,ycm终于可以补全出来struct addrinfo这个结构体的成员了~

  这样,问题就一定在ubuntu上的/usr/include/netdb.h这个文件了,再回头看这个文件的struct addrinfo的部分:

  

  而mac上的/usr/include/netdb.h的文件先关部分如下:

  

  对,二者的产别就是红框框的部分 #ifdef __USE_POSIX

  到此,基本可以确定,就是因为多了ifidef,导致ycm的补全机制直接漏过了struct addrinfo这个结构体

  但是又有一个矛盾的事情:既然ycm补全机制把这个strut addrinfo错过了,为什么在ubuntu上用gcc编译链接的程序还能正确执行结果呢?证明只是ycm漏过了这个struct addrinfo,但是gcc缺没有漏过。

  又想起来,ycm后台用的clang编译器相关的内容,而不是gcc。既然ubuntu的文件既然能被gcc正确编译链接,可以判定,还是问题出在了ycm的配置本身。但是,这又回到之前的思路二,似乎又没有什么办法了。

  也是偶然,我试验能不能补全另一个结构体。试验/usr/include/netinet/in.h中的结构体,居然能够ycm补全出成员变量:

  

  这就证明了,ycm的编译一定是没有问题的,问题一定出在ycm的配置了,并且跟idef __USE_POSIX有关。

  于是,走出了解决问题的一步:直接google “ubuntu #ifdef __USE_POSIX”:

http://stackoverflow.com/questions/12024703/why-cant-getaddrinfo-be-found-when-compiling-with-gcc-and-std-c99

  通过这个帖子,知道了问题出在了.ycm_extra_conf.py配置文件中-std=c99这个上面,改为gnu99,问题就解决了。如图:

  

  问题大概就是这样解决了。

  总之,-std=c99是标准的c,如果把这个喂给clang的配置,则clang就会忽略“#ifdef __USE_POSIX”这种的定义,因为这属于c extension,不属于c99的范畴;因此改为-std=gnu99就OK了。搞笑的是,clang的默认-std是gnu99,结果在这里我画蛇添足,导致了后面的各种问题。

  这么看来,gcc编译器的默认选项也是-std=gnu99,至少肯定是支持一些c extrension的。否则,程序也不可能正确执行出结果,gcc编译链接的过程就报错了。

  如果还想再多了解些c的标准和编译器的问题,可以看看这个日志:

  http://www.crifan.com/summary_c_language_version_c89_amd1_c99_c11/

  

  另,ycm的自动语法检查暂时有些碍事儿,将其关掉的方法查了一下:http://stackoverflow.com/questions/24500281/youcompleteme-and-syntastic-compatibility

  添加了ycm的跳转配置:gb跳转到定义或者声明,ctrl+o跳转回去。

【vim环境配置】解决ubuntu上 由YouCompleteMe插件配置不当引起的 自动补全失效的问题的更多相关文章

  1. xcode4 语法高亮和自动补全失效的解决办法

    xcode4 语法高亮和自动补全失效的解决办法xcode4有一个bug,某些时候会造成类名的着色显示不出来,还有就是代码补全不能完全显示,而且没有参数. 解决方法: 首先关闭项目窗口,然后到Organ ...

  2. ubuntu1204上不能正常用emacs配合gocode进行自动补全

    我按gocode的页面https://github.com/nsf/gocode上去做,可是还是未成功,,我确认auto-complete在c-mode中是可以使用的,因为有补全出来了, 我再找了ht ...

  3. vnc xfce tab自动补全失效的解决方法

    edit~/.config/xfce4/xfconf/xfce-perchannel-xml/xfce4-keyboard-shortcuts.xml find the line <proper ...

  4. Ubuntu eclipse 命令补全失效 (转载)

    我的eclipse 3.4,从ibm网站上下载解压后使用.发觉自动补全功能(alt + /)失效. 解决的办法: 1.(eclipse)window --> preferences --> ...

  5. VIM中使用tab键自动完成(vim tab键自动补全 )插件supertab

    supertab.vmb 这个插件好好用, Tab自动补全 http://www.vim.org/scripts/script.php?script_id=1643 安装步骤: 1.下载 supert ...

  6. 如何配置webpack让浏览器自动补全前缀

    一.postcss-loader有什么用? PostCSS 本身是一个功能比较单一的工具.它提供了一种方式用 JavaScript 代码来处理 CSS.它负责把 CSS 代码解析成抽象语法树结构(Ab ...

  7. Linux实战(12):解决Centos7 docker 无法自动补全

    环境:centos最小化安装,会出现一些命令无法自动补全的情况,例如在docker start 无法自动补全 start 命令,无法自动补全docker容器名字.出现这种情况的可参考以下操作: yum ...

  8. Vim自动补全神器YouCompleteMe的配置

    简介:YouCompleteMe号称Vim的自动补全神器,该项目在github的地址:YouCompleteMe:以下在10.0.1 build-1379776平台配置完成 插件安装操作: 1.确保V ...

  9. Vim+Taglist+AutoComplPop之代码目录分栏信息和自动补全提示(Ubuntu环境)

    一步: 首先在Ubuntu环境中安装ctags:  sudo apt-get install ctags 第二部:       安装Taglist-------------Taglist是vim的一个 ...

随机推荐

  1. android 中组件继承关系图,一目了然

    View继承关系图 Adapter适配器继承关系图 Activity继承关系图

  2. L1 loss L2 loss

    https://www.letslearnai.com/2018/03/10/what-are-l1-and-l2-loss-functions.html http://rishy.github.io ...

  3. spring boot Unable to find a @SpringBootConfiguration, you need to use @ContextConfiguration

    java.lang.IllegalStateException: Unable to find a @SpringBootConfiguration, you need to use @Context ...

  4. vue快速使用

    1.引用脚本 <script src="https://cdn.jsdelivr.net/npm/vue@2.5.17/dist/vue.js"></script ...

  5. 【洛谷P1108】低价购买

    低价购买 题目链接 n<=5000 n^2的算法是可以接受的 第一个数字显然是求最长下降子序列,可以n^2或nlognDP 要求方案数,可以在n^2算法中做一些修改,DP求方案数 dp[i]表示 ...

  6. Android学习笔记_78_ Android开发中使用软引用和弱引用防止内存溢出

    在<Effective Java 2nd Edition>中,第6条“消除过期的对象引用”提到,虽然Java有 垃圾回收机制,但是只要是自己管理的内存,就应该警惕内存泄露的问题,例如的对象 ...

  7. 封装方法到对象(javascript)

    /*! * artDialog 5 * Date: 2012-03-21 * http://code.google.com/p/artdialog/ * (c) 2009-2012 TangBin, ...

  8. 通过sql语句查询出来的结果字段没有到对应实体类时的处理方法

    通过sql语句查询出来的结果字段没有到对应实体类时的处理方法,对于Person类获取用户第一个名字和年龄两个字段,常见的有两种方式: 1.在创建一个与查询结果字段对应的类,然后通过构造函数实现: Qu ...

  9. js 3秒后跳转页面的实现代码

    隔多少秒后自动跳转到其它页(js脚本) 方法一: $(function(){ Load(URL); }) var secs = 3; //倒计时的秒数 var URL = "<?= u ...

  10. Vue nodejs商城-订单模块

    一.订单列表渲染 新建OrderConfirm.vue订单确认页面,添加路由 src/router/index.js添加路由 import OrderConfirm from '@/views/Ord ...