用python实现一个按需生成用于vim跳转的tags文件的小程序

初学python,找个有用的来练手。选定了一个我早就想做的让vim按需生成tags跳转文件的小程序。

vim的tags用于跳转的好处自是不用多说,但是如何生成tags确实困扰我的一大问题,用VS的都知道,VS会为每个解决方案生成一个ncb文件用于自动提示,相当于tags的功能,这个ncb文件很智能,它会根据你所包含的头文件来自动处理。这样,你只需包含一个头文件,就可以出现该头文件相关的自动提示了。

但是vim就没有像VS这么智能了,目前我的做法就是在某个目录下调用ctags,然后加上递归参数,这样的缺点就是会生成一个很大的tags文件,里面很多都是项目根本没用到的头文件的tags。为了改善这种情况,需要有一个按所用到的头文件来生成tags的方法,让ctags按需生成tags。

ctags本身是支持从一个文件中读入需要生成tags的文件列表的,因此剩下的问题就是如何找到项目所用到的所有相关的C++文件,然后让ctags从这些文件中生成tags。这就是我拿来练手的小程序了。

python的好处就是不用像C++那样考虑那么多不相关的细节,专注于你想做的。代码如下,应该还算精简吧。有更好的方法不妨评论分享下,:)

# -*- coding: utf8 -*-  
# by absolute8511 2010-12-04
import os,sys
from os import path
import string,re

#在系统头文件目录下查找系统头文件,可以自行添加头文件路径
def findsysinclude(filename,includepath):
    if filename==None:
        return None
    sysincludepath = ['/usr/include/','/usr/include/c++/4.4/',
            '/usr/lib/gcc/i486-linux-gnu/4.4/include/','/usr/include/c++/4.4/i486-linux-gnu/']   #系统头文件目录
    includepath += sysincludepath
    for onepath in includepath:
        if path.isfile(onepath+filename):
            return onepath+filename
    print 'Not found: '+filename
    return None

#读取用户传入的头文件路径参数
args = sys.argv
print args
retest = re.compile(r'\.cpp$|\.c$|\.h$|\.hpp$|\.cc$',re.IGNORECASE)
cppfiles = []
#查找用户项目路径下的所有源文件
for root,dirs,files in os.walk(os.getcwd()):
    oldlen = len(cppfiles)
    cppfiles += filter(lambda x:retest.search(x)<>None,files)        #过滤非C/C++文件
    for index in range(oldlen,len(cppfiles)):
        cppfiles[index] = root + '/' + cppfiles[index];

oldstdout = sys.stdout
sys.stdout= logfile = open('maketags.log','w')

print 'total cpp files: ' + str(len(cppfiles))
includetest = re.compile(r'\s*#include\s*((<(?P<sysincludefile>.*?)>)|("(?P<userincludefile>.*?)"))')  ## *?代表不贪婪匹配

#对每个C++源文件,查找它包含的头文件以及它的头文件所包含的头文件,以次递归直到全部包含
#由于已经遍历过用户目录了,因次只需要再把系统头文件加近来即可
for cppfile in cppfiles:
    try:
        fp = open(cppfile,'r')
        print 'finding in file: '+cppfile
        for line in fp:
            result = includetest.search(line)   #正则匹配头文件
            if result <> None:
                print 'Matched include line is:'+result.group()
                tmp = findsysinclude(result.group('sysincludefile'),args[1:])
                tmp2 = findsysinclude(result.group('userincludefile'),args[1:]) #用户头文件也去找系统目录
                if tmp<>None and cppfiles.count(tmp)==0:
                    cppfiles += [tmp]
                    print 'Added sys files: '+tmp
                # tmp2 = path.abspath(path.dirname(cppfile)+'/'+result.group('userincludefile'))
                if tmp2<>None and cppfiles.count(tmp2)==0:
                    cppfiles += [tmp2]
                    print 'Added user files: '+tmp2
    finally:
        fp.close()
try:
    filelist = open('python_tags_filelist','w')
    filelist.writelines('\n'.join(cppfiles))
finally:
    filelist.close()

print 'total cpp and include files:'+str(len(cppfiles))
# print 'after delete duplicate files: ' + str(len(set(cppfiles)))
sys.stdout = oldstdout
logfile.close()

os.system('ctags --c++-kinds=+p --fields=+iaS --extra=+q -L python_tags_filelist')

posted on 2010-12-04 22:57  absolute  阅读(1942)  评论(0编辑  收藏  举报

导航