程序员功能

前面我们用了5讲的篇幅来讲基本编辑的基本功:第4讲是基本操作,第5讲是操作符,第6讲行编辑ex命令,第7讲可视模式,第8讲多窗口,第9讲缓冲区和标签页。

从这一讲开始,我们从通用功能向程序员常用功能挺进。首先我们从浏览代码最常用的跳转功能开始。

代码跳转

代码跳转,需要传说中的Exuberant Ctags工具,地址在:http://ctags.sourceforge.net/

当然大家根据系统不同各显神通吧。比如我是macOS,就通过Homebrew来安装了。大家手头有什么”apt install”,”dnf install”,”zypper install”之类的独门兵器就往上用吧。

目前Exuberant ctags支持41种编程语言,具体列表在:http://ctags.sourceforge.net/languages.html

如果要支持这41种语言之外的,请看扩展指南:http://ctags.sourceforge.net/EXTENDING.html

将来我们用到了再说,目前常用的语言要么是ctags已经支持了,要么是语言官方有支持了。基本上常用语言对于vim和emacs的支持还是比较全的。

代码跳转速成教程

  • 首先我们用vim打开源代码的根目录。比如我打开Google的Magenta内核的源代码根目录

  • 运行ctags重新生成索引
:!ctags -R

源代码有较大变化后,就需要重新生成索引,否则索引就匹配不上了。

  • 设置要查找的tags文件的路径

    先可以通过:set tags?来查查当前值
:set tags?

默认值为:

tags=./tags,tags

如果想要设置的话,可以用:set tags=tags文件名的方式来设置。

我们因为是在源码根目录下刚生成,应该是能找到的,我们就直接进入下一步。

  • 为了对照方便。我们可以充分利用之前学习的多窗口功能。比如用:vsplit竖切成两半,对照着看。

  • Ctrl-] 跳转。比如我想查看mp_init()是实现啥的,光标定位过去之后,点击”Ctrl-]”,就跳转到mp_init实现的部分:

  • 如果要退回刚才的位置,功能键是Ctrl-t

  • 如果想让左右屏的内容重新同步一下,把左边关了再重切就是了

:only | :vsplit

如果有重名怎么办?

对于没有重名的,上面的方式都工作得很好。但是很不幸,有时候,函数的定义是有重名的。这时候我们不方便简单地用Ctrl-]了,我们有多选的g+Ctrl-]. 如图的例子:arch_mp_send_api函数在arm64和x86上各有自己的实现,我们用g+Ctrl-]就会出现下面的选择列表:

不是非要跳到代码上才能查

如果跳到代码上查比较慢的话,我们也可以通过ex命令直接搜。

  • :tag {关键字} - 相当于Ctrl-]
  • :tjump {关键字} - 相当于g+Ctrl-]

除了全字匹配,:tag和:tjump命令还支持正则表达式搜索。

比如,用:tjump init,全字匹配能得到的结果如下:

  # pri kind tag               file
  1 F   m    init              kernel/include/app.h
               struct:app_descriptor
               app_init  init;
  2 F   m    init              kernel/lib/unittest/include/unittest.h
               struct:unitest_testcase_registration
               unitest_testcase_init_fn_t      init;
  3 F   m    init              system/ulib/ddk/include/ddk/driver.h
               struct:mx_driver_ops
               mx_status_t (*init)(void** out_ctx);
  4 F   m    init              system/ulib/mxio/private.h
               struct:__anon440
               bool init;
  5 F   f    init              third_party/uapp/dash/src/init.c
               init() {
  6 F   v    init              third_party/uapp/dash/src/mkinit.c
               char init[] = "\
  7 F   m    init              third_party/ulib/cryptolib/include/lib/crypto/cryptolib.h
               struct:clHASH_vtab
               void (* const init)(struct clHASH_CTX*);
  8 F   f    init              third_party/ulib/qrcodegen/qrcode.cpp
               class:qrcodegen::ReedSolomonGenerator
               Error ReedSolomonGenerator::init(size_t degree) {
  9 FS  f    init              system/core/acpisvc/main.c
               static ACPI_STATUS init(void) {
 10 FS  f    init              system/uapp/gpt/gpt.c
               static gpt_device_t* init(const char* dev, bool warn, int* out_fd) {
 11 FS  f    init              third_party/uapp/dash/src/mksyntax.c
               init(void)

我们换成:tjump /init,则包含init所有的内容都能搜出来,要显示好多屏:

  # pri kind tag               file
  1 F   m    init              kernel/include/app.h
               struct:app_descriptor
               app_init  init;
  2 F   m    init              kernel/lib/unittest/include/unittest.h
               struct:unitest_testcase_registration
               unitest_testcase_init_fn_t      init;
  3 F   m    init              system/ulib/ddk/include/ddk/driver.h
               struct:mx_driver_ops
               mx_status_t (*init)(void** out_ctx);
  4 F   m    init              system/ulib/mxio/private.h
               struct:__anon440
               bool init;
  5 F   f    init              third_party/uapp/dash/src/init.c
               init() {
  6 F   v    init              third_party/uapp/dash/src/mkinit.c
               char init[] = "\
  7 F   m    init              third_party/ulib/cryptolib/include/lib/crypto/cryptolib.h
               struct:clHASH_vtab
               void (* const init)(struct clHASH_CTX*);
  8 F   f    init              third_party/ulib/qrcodegen/qrcode.cpp
               class:qrcodegen::ReedSolomonGenerator
               Error ReedSolomonGenerator::init(size_t degree) {
  9 FS  f    init              system/core/acpisvc/main.c
               static ACPI_STATUS init(void) {
 10 FS  f    init              system/uapp/gpt/gpt.c
               static gpt_device_t* init(const char* dev, bool warn, int* out_fd) {
 11 FS  f    init              third_party/uapp/dash/src/mksyntax.c
               init(void)
 12     f    Init              kernel/arch/x86/hypervisor.cpp
               class:PerCpu
               status_t PerCpu::Init(const VmxInfo& info) {
 13     f    Init              kernel/arch/x86/hypervisor.cpp
               class:VmcsPerCpu
               status_t VmcsPerCpu::Init(const VmxInfo& vmx_info) {
 14     f    Init              kernel/dev/pcie/pcie_bridge.cpp
               class:PcieBridge
               status_t PcieBridge::Init(PcieUpstreamNode& upstream) {
 15     f    Init              kernel/dev/pcie/pcie_device.cpp
               class:PcieDevice
               status_t PcieDevice::Init(PcieUpstreamNode& upstream) {
 16     f    Init              kernel/kernel/vm/vm_aspace.cpp
               class:VmAspace
               status_t VmAspace::Init() {
 17     f    Init              kernel/lib/magenta/io_mapping_dispatcher.cpp
               class:IoMappingDispatcher
               status_t IoMappingDispatcher::Init(const char* dbg_name,
 18     f    Init              kernel/lib/mxtl/arena.cpp
               class:mxtl::Arena
               status_t Arena::Init(const char* name, size_t ob_size, size_t count) {
 19     f    Init              kernel/lib/mxtl/arena.cpp
               class:mxtl::Arena::Pool
               void Arena::Pool::Init(const char* name, mxtl::RefPtr<VmMapping> mapping,
 20     f    Init              system/dev/bus/virtio/block.cpp
               class:virtio::BlockDevice
               mx_status_t BlockDevice::Init() {
 21     f    Init              system/dev/bus/virtio/gpu.cpp
               class:virtio::GpuDevice
               mx_status_t GpuDevice::Init() {
 22     f    Init              system/dev/bus/virtio/ring.cpp
               class:virtio::Ring
               mx_status_t Ring::Init(uint16_t index, uint16_t count) {
-- More --

小结

  • !ctags -R 生成索引
  • :set tags=路径 设定ctags生成的tags文件的路径
  • Ctrl-] 跳转到唯一的或者是第一个匹配的代码
  • g+Ctrl-] 提供多选列表
  • :tag {关键字} : ex命令版的Ctrl-]
  • :tjump {关键字}: ex命令版的g+Ctrl-]
  • :tag或:tjump /{关键字} :正则表达式搜索

Vim技能修炼教程(10) - 代码跳转的更多相关文章

  1. Vim技能修炼教程(11) - 代码折叠

    上一讲我们是程序员篇的第一讲,关于代码跳转.代码跳转是一个付出很少收获很大的功能.这一节我们开始一个收获很多,但是付出也相对多一点功能:代码折叠. 代码折叠 折叠的类型 折叠有下面几种类型: * Ma ...

  2. Vim技能修炼教程(2) - 语法高亮速成

    语法高亮速成 我们继续在人间修行Vim技能之旅.上一次我们学习了如何通过vundle安装插件,这次我们迅速向写插件的方向挺进. 我们先学习一个最简单的语法高亮插件的写法. 语法高亮基本上是由三部分组成 ...

  3. Vim技能修炼教程(13) - 变量

    VimScript变量 上节我们介绍了Python和Ruby来编写Vim插件的方式. 不过,Python和Ruby并不是所有的Vim都支持的功能,如果以最小依赖的原则来说,还是原汁原味的Vimscri ...

  4. Vim技能修炼教程(8) - 多窗口

    多窗口 如果一个vim只能开一个窗口,那肯定是有点low.尤其是写代码的时候,打开多个文件是经常的需求. 速成教程 横着切成两个 :split 文件名 上下切换窗口 Ctrl-W加上上下键,可以实现上 ...

  5. Vim技能修炼教程(7) - 可视模式

    可视模式 可视模式是与正常模式.插入模式一起并列的模式.它的作用就像图形化编辑器下用鼠标来选择一个块. 在vim下,使用正常模式和ex命令,连搜带跳行的,未必就比用鼠标慢. 我们先做一个例子找找感觉, ...

  6. Vim技能修炼教程(4) - 基本功

    基本功 前面我们学会了插件管理器和如何实现语法高亮,相信大家已经从中体会到了vim插件的强大功能.现在,是时候回来补一补基本功了. Vi有三种主要模式,正常模式,插入模式和可视化模式.正常我们推荐的方 ...

  7. Vim技能修炼教程(3) - 语法高亮进阶

    语法高亮进阶 首先我们复习一下上节学到的三个命令: * syntax match用于定义正则表达式和规则的对应 * highlight default定义配色方案 * highlight link将正 ...

  8. Vim技能修炼教程(1) - 使用vundle管理插件

    世界上有两个伟大的编辑器:一个是emacs,一个是vi.它们从诞生以来,一直在Unix/Linux世界得到最广泛的支持. 尽管过了几十年,在Windows平台上和跨平台上有层出不穷的后起之秀不断挑战它 ...

  9. Vim技能修炼教程(16) - 浮点数计算函数

    浮点数计算函数 这一节的所有函数,只有在vim编译时支持了+float时才有效. 三角函数 sin() : sine正弦函数 cos() : cosine余弦函数 tan() : tangent正切函 ...

随机推荐

  1. 20145307第三次JAVA学习实验报告

    20145307 <Java程序设计>第三次实验报告 北京电子科技学院(BESTI)实验报告 课程:Java程序设计 班级:1453 指导教师:娄嘉鹏 实验日期:2016.04.22 实验 ...

  2. LeetCode——maximal-rectangle

    Question Given a 2D binary matrix filled with 0's and 1's, find the largest rectangle containing all ...

  3. unix_timestamp() 和 from_unixtime()

    unix_timestamp() 将时间转换为时间戳.(date 类型数据转换成 timestamp 形式整数) select unix_timestamp('2016-03-23 11:10:10' ...

  4. 解题报告:hdu 1005 number subsequent

    2017-09-06 20:35:59 writer:pprp 本来以为这是一道水题,写了一个递归就赶紧交上去了, 结果超时了,看看数据范围100000000,肯定把栈给爆了 想用记忆化的方法,但是虽 ...

  5. 简单描述DataAdapter、DataReader、DataSet、Datatable对比

    一.存储的对比 DataReader 从数据库中检索[只读]数据流,存在客户端网络缓冲区,直到Read方法访问它们. DataAdapter 表示一组SQL命令和数据库连接,用于填充DateSet和[ ...

  6. RabbitMQ入门(4)——路由(Routing)

    这一篇我们将介绍如何订阅消息的一个子集.例如,我们只需要将日志中的error消息存储到日志文件中而将所有日志消息都在控制台打印出来. 绑定(Bindings) 在前面的例子中,我们创建了交换机和队列的 ...

  7. Python性能(转)

    第一部分 阅读 Zen of Python,在Python解析器中输入 import this. 一个犀利的Python新手可能会注意到"解析"一词, 认为Python不过是另一门 ...

  8. 使用阿里云ECS安装HDFS的小问题

    毕设涉及HDFS,理论看的感觉差不多了,想搭起来测试一下性能来验证以便进行开题报告,万万没想到装HDFS花费了许多天,踩了许多坑,记录一下. 背景:使用两台阿里云学生机ECS,分处不同账号不同区域,一 ...

  9. PHP中用下划线开头的含义

    命名的规则 加一个为私有的 加两个一般都是系统默认的,系统预定义的,即所谓:=====================“魔术方法”与“魔术常量”=====================★PHP起止为 ...

  10. 欢迎来到 Flask 的世界

    欢迎来到 Flask 的世界 欢迎阅读 Flask 的文档.本文档分成几个部分,我推荐您先读 < 安装 >,然后读< 快速上手 >.< 教程 > 比快速上手文档更详 ...