tags 在使用vim編程和瀏覽代碼是非常有用。可以用CTRL+]和CTRL+t 來回跳轉關鍵字。
先生成自己工作目錄的tags。最簡單粗暴用法:
$cd yourwork $ctags -R *
這樣會生成一個tags文件。(注:對於gvim的話,需要進入命令提示符下,cd到工作目錄,來產生tags文件,直接在gvim去生成的話,會出現tags生成在上一級目錄的情況)
不過,這種有個問題,成員變量沒有包含在里面。所以自動完成對象的成員時沒有提示。
解決辦法:
$ctags -R --fields=+iaS --extra=+q *
–fields=[+|-]flags
–fields指定tags的可用擴展域(extension fields),以包含到tags入口。
i:繼承信息Inheritance information
a:類成員的訪問控制信息 Access (or export) of class members
S: 常規簽名信息,如原型或參數表 Signature of routine(e.g. prototype or parameter list)
–extra=[+|-]flags
指定是否包含某種擴展信息到tags入口。
q:包含類成員信息(如c++,java,Eiffel)。
但就算是C 語言的結構,也需要這兩個參數設置才能獲取成員信息。
這樣就能自動完成結構和類的成員了。
但是,對於系統的函數,還是沒有跳轉。如socket定義,inetaddr_in這樣的結構沒有自動變量完成。
最簡單做法:
$ctags --fields=+iaS --extra=+q -R -f ~/.vim/systags /usr/include /usr/local/include
然后在.vimrc里設置
set tags+=~/.vim/systags
這樣雖然基本能跳轉到系統函數定義,一個問題是某些系統函數並沒有加入到systags里。
如/usr/incluce/socket.h的socket系列函數,memset等很多關鍵函數都沒有到tag里:
extern int listen (int __fd, int __n) __THROW;
這是因為 __THROW的宏定義讓ctags不再認為該系列函數是函數。
同理,如memcpy系列函數:
如/usr/include/string.h的
extern int strcmp (__const char *__s1, __const char *__s2) __THROW __attribute_pure__ __nonnull ((1, 2));
還有attribute_pure ,nonull等屬性,都需要忽略。如果需要#if 0里面的定義,可以–if0=yes來忽略 #if 0這樣的定義。
$ctags -I __THROW -I __attribute_pure__ -I __nonnull -I __attribute__ --file-scope=yes --langmap=c:+.h --languages=c,c++ --links=yes --c-kinds=+p --c++-kinds=+p --fields=+iaS --extra=+q -R -f ~/.vim/systags /usr/include /usr/local/include
這樣.vim/systags里面是全的,但內容過多。一個函數定義的跳轉,會有幾十個候選。這時我們可以簡化一下,將-R去掉,自己指定目錄:
$ctags -I __THROW -I __attribute_pure__ -I __nonnull -I __attribute__ --file-scope=yes --langmap=c:+.h --languages=c,c++ --links=yes --c-kinds=+p --c++-kinds=+p --fields=+iaS --extra=+q -f ~/.vim/systags /usr/include/* /usr/include/sys/* /usr/include/bits/* /usr/include/netinet/* /usr/include/arpa/* /usr/include/mysql/*
還可以包含一些自己編程需要的路徑。注意后面加*號。
這樣生成的系統tags就少多了。不會有太多不相干的定義。
另外在vim中將光標移動到#include <xxx.h>頭文件名中,按esc,gf(助記詞:goto file),即可查看頭文件