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

摘要: Vimscript的变量、列表和字典类型

VimScript变量

上节我们介绍了Python和Ruby来编写Vim插件的方式。
不过,Python和Ruby并不是所有的Vim都支持的功能,如果以最小依赖的原则来说,还是原汁原味的Vimscripts是放置四海Vim而皆灵的方式。当代码规模变大时,Python,Ruby,Perl这些语言的引入将带来较高的效率。但是Vimscripts仍然是最基本的Vim语言,值得我们首先学好。

变量

做为一种脚本语言,Vimscript当然是支持变量的。
不过Vimscript的特色是,显式指定作用域。也就是说,一个变量,需要在名字前加作用域前缀。

这些作用域前缀有:

  • b: 缓冲区级作用域
  • w: 窗口级作用域
  • t: 标签页级作用域
  • g: 全局变量
  • l: 函数内局部变量
  • s: vim脚本文件级作用域
  • a: 函数参数
  • v: Vim专用的全局变量

如果未指定前缀,定义于函数内部的默认为l:,而定义于函数之外的默认为g:.

Vimscript通过:let命令来为变量赋值,赋值之前不需要定义。

访问属性和寄存器

Vimscript通过&属性名的方式来访问属性,也就是除了:set之外,我们也可以通过:let为变量赋值。

同变量一样,属性也有作用域,可以通过&l:属性来只改本地属性。相当于我们使用了:setlocal命令。

:let还可以通过@来访问寄存器。
还记得吗?d,c,y等命令支持将删除或复制的内容放到寄存器中,如果未指定,则放入""寄存器中。"0到"9中最储历史值。还有"a到"z一共26个命名寄存器。

数据类型

首先是数据类型,Vimscript提供下面9种基本数据类型:

  • Number: 32位带符号整数. 如果支持+num64,就是64位带符号整数。支持十六进制,二进制和八进制。
  • Float: 浮点数。只有编译时支持+float才有。
  • String: 字符串。
  • List: 有序列表。
  • Dictionary: 哈希表。
  • Funcref: 函数引用。
  • 特殊值
    • v:false
    • v:true
    • v:none
    • v:null
  • Job: 任务相关
  • Channel: 通道相关

一个变量的类型可以通过type()函数查看。

数字和字符串

在Vimscript中,字符串和数字之间会进行自动转换。我们看下官方的几个例子:

  • String "456" --> Number 456
  • String "6bar" --> Number 6
  • String "foo" --> Number 0
  • String "0xf1" --> Number 241
  • String "0100" --> Number 64
  • String "0b101" --> Number 5
  • String "-8" --> Number -8
  • String "+8" --> Number 0

"foo"被转换成0,和"+8"被转换成0这两条case要特殊注意。

另外,八进制"0100"的情况也要特别注意。
为了避免出现八进制转换的问题,我们可以使用str2nr函数来做显示转换。因为str2nr默认是10进制的。

echo 0100

的结果为64

echo str2nr("0100")

的值为100.

真值和假值

在Vimscript中,0表示假值,其余数值均表示真值。假值也可以用v:false表示,真值为v:true.

比如"foo"的值为假,因为"foo"转换成整数的值是0.

列表类型

Vimscript的列表可以混杂各种元素。

例:

:let b:list1 = [1, 4.1e+3, "Hello",[1,2]]

列表可以用下标访问。比如b:list1[0].
列表的下标从0开始计数。而[-1]表示最后一个元素。

如果要给列表元素一个不存在时的默认值的话,可以使用get()函数。

例:

        :echo get(mylist, idx)
        :echo get(mylist, idx, "NONE")

两个列表可以通过+号来连接成一个列表:

        :let longlist = mylist + [5, 6]
        :let mylist += [7, 8]

子列表

Vimscript取子列表跟其它脚本语言很像,用[起始位置:结束位置]来进行切片。

        :let shortlist = mylist[2:-1]   " get List [3, "four"]

如果是从第一个元素开始,或者到最后一个元素结束,都可以省略:

        :let endlist = mylist[2:]       " from item 2 to the end: [3, "four"]
        :let shortlist = mylist[2:2]    " List with one item: [3]
        :let otherlist = mylist[:]      " make a copy of the List

如果越界了,也不会产生错误,到最后一个元素结束了就是了么。

        :let mylist = [0, 1, 2, 3]
        :echo mylist[2:8]               " result: [2, 3]

列表的赋值和拷贝

如果将一个列表赋给另一个变量,不会复制列表,只是传递引用。
如果想要复制一份列表的话,需要使用copy函数。
例:

        :let aa = [[1, 'a'], 2, 3]
        :let bb = copy(aa)
        :call add(aa, 4)
        :let aa[0][1] = 'aaa'
        :echo aa
        [[1, aaa], 2, 3, 4]
        :echo bb
        [[1, aaa], 2, 3]

但是请注意,copy只是浅copy,如果有子列表,是只复制引用的。如果真的要大搬家,需要使用deepcopy()函数。

列表的比较

  • is: 比较两个变量是否引用同一列表
  • ==:比较两个列表的值是否相等

另外需要注意的是,对于变量,会自动将字符串转成数字,但是对于列表中的元素,没有这种自动转换。

        echo 4 == "4"
        1
        echo [4] == ["4"]
        0

列表的多变量赋值

如果要引用列表中的值,可以同时将其赋给多个变量。
例:

        :let [var1, var2; rest] = mylist

等价于:

        :let var1 = mylist[0]
        :let var2 = mylist[1]
        :let rest = mylist[2:]

这个对于函数返回值之类的是个利好。

列表的修改

对于可以定位的值,直接通过:let赋值的方式修改。

对于插入,添加,删除等,使用相关函数来进行操作:

        :call insert(list, 'a')         " prepend item 'a'
        :call insert(list, 'a', 3)      " insert item 'a' before list[3]
        :call add(list, "new")          " append String item
        :call add(list, [1, 2])         " append a List as one new item
        :call extend(list, [1, 2])      " extend the list with two more items
        :let i = remove(list, 3)        " remove item 3
        :let l = remove(list, 3, -1)    " remove items 3 to last item
        :call filter(list, 'v:val !~ "x"')  " remove items with an 'x'

列表还可以排序、反转和排重:

        :call sort(list)                " sort a list alphabetically
        :call reverse(list)             " reverse the order of items
        :call uniq(sort(list))          " sort and remove duplicates

使用for循环来处理列表

for循环是处理列表的最简用手段:

        :for item in mylist
        :   call Doit(item)
        :endfor

for循环支持解构式的赋值:

        :for [i, j; rest] in listlist
        :   call Doit(i, j)
        :   if !empty(rest)
        :      echo "remainder: " . string(rest)
        :   endif
        :endfor

列表常用函数

  • empty函数:判空
  • len函数:计算列表元素个数
  • max函数:求列表最大元素
  • min函数:求列表最小元素
  • count函数:计算元素在列表中出现的次数
  • index函数:求元素第一次出现的位置
  • split函数:将字符串拆分成列表
  • join函数:将列表连接成一个字符串

字典

Vimscript的字典功能,其实就是哈希表。

例:

        :let mydict = {1: 'one', 2: 'two', 3: 'three'}
        :let emptydict = {}

访问字典可以使用[]:

        :let val = mydict["one"]
        :let mydict["four"] = 4

在没有歧义的情况下,也可以使用.:

        :let val = mydict.one
        :let mydict.four = 4

字典遍历

可以通过遍历keys列表和values列表的方式来遍历字典。

例:

        :for key in keys(mydict)
        :   echo key . ': ' . mydict[key]
        :endfor
        :for v in values(mydict)
        :   echo "value: " . v
        :endfor

还可以遍历前先排一次序:

        :for key in sort(keys(mydict))

也可以将key和value打包在一起,使用item函数遍历:

        :for [key, value] in items(mydict)
        :   echo key . ': ' . value
        :endfor

字典的复制

字典的复制也同样要使用copy和deepcopy函数。

字典的增删改查

字典的修改和增加都很简单,直接赋值就是了:

        :let dict[4] = "four"
        :let dict['one'] = item

用云栖社区APP,舒服~

原文链接

posted @ 2017-07-17 15:00  暖夏未眠丶  阅读(1039)  评论(0编辑  收藏  举报