vim有各種強大的插件,這不僅歸功于其提供的用來編寫插件的腳本語言vimL,還得益于它良好的接口實現,從而支持python等語言編寫插件。當vim編譯時帶有
+python
特性時就能使用python2.x編寫插件,
+python3
則支持python3.x,可以使用
vim --version
來查看vim的編譯特性。
要使用python接口,可以用
:h python
來查看vim提供的幫助文檔,本文做一個簡單的介紹。我們都知道在vim里可以執行bash命令,只需要
:!command
即可,那么vim里可以執行python語句嗎?當然可以了,vim那么強大!不是嗎,是嗎?!
vim中執行python命令
在vim中可以使用
py[thon] {stmt}
來執行python語句{stmt},你可以用
:python print "Hello World!"
來驗證一下。
只能執行一條語句,沒什么用,不是嗎?所以有更加強大的接口,語法如下:
py[thon] << {endmarker}
{script}
{endmarker}
這樣我們就可以執行python腳本{script}中的內容了。{endmarker}是一個標記符號,可以是任何內容,不過{endmarker}后面不能有任何的空白字符??匆粋€簡單的例子,假設下面代碼保存為script_demo.vim:
function! Foo()
python << EOF
class Foo_demo:
??? def __init__(self):
??????? print 'Foo_demo init'
Foo_demo()
EOF
endfunction
那么在vim中我們先用
:source path_to_script/script_demo.vim
來加載腳本,然后就可以用
:call Foo()
來運行python腳本了,整個過程如圖所示:
此外,我們還可以將python腳本放到一個單獨的.py文件中,然后用
pyf[ile] {file}
來運行python文件中的程序,要注意這里pyf[ile]后面的所有參數被看做是一個文件的名字。
vim模塊
我們已經可以在vim中執行python命令了,但是python怎么獲取vim中的一些信息呢?比如說我想知道vim當前緩沖區一共有多少行內容,然后獲取最后一行的內容,用python該怎么做呢?
于是vim提供了一個python模塊,有趣的是模塊名字就叫做vim,我們可以用它來獲取vim編輯器里面的所有信息。上面問題用以下python腳本就可以解決了:
function! Bar()
python << EOF
import vim
cur_buf = vim.current.buffer
print "Lines: {0}".format(len(cur_buf))
print "Contents: {0}".format(cur_buf[-1])
EOF
endfunction
你可以自己加載腳本運行一下見證奇跡!上面代碼出現了
vim.current.buffer
,想必你已經從名字猜到了它的意思了,不過還是來詳細看下吧:
vim模塊中的常量
vim.buffers: 用來訪問vim中緩沖區的列表對象,可以進行如下操作:
:py b = vim.buffers[i]??? # Indexing (read-only)
:py b in vim.buffers????? # Membership test
:py n = len(vim.buffers)? # Number of elements
:py for b in vim.buffers: # Iterating over buffer list
vim.windows: 用來訪問vim中窗口的列表對象,和vim.buffers支持的操作基本相。
vim.current: 用來訪問vim中當前位置的各種信息,比如:
vim.current.line
vim.current.buffer
vim.current.window
vim.current.tabpage
vim.current.range
vim.vvars: 類似字典的對象,用來存儲global(g:)變量或者vim(v:)變量。
還有其他的一些常量,這里不做敘述。注意這里的常量并不是真正意義上的常量,你可以重新給他們賦值。但是我們應該避免這樣做,因為這樣會丟失該常量引用的值?,F在為止我們已經能獲取vim中數據,然后用python來對其進行操作,似乎完美了。
不過vim并沒有止步于此,它可是
Stronger than Stronger
!因為我們可以在python里使用vim強大的命令集,這樣就可以用python寫一些常用的批處理插件,看下面簡單的例子:
function! Del(number)
python << EOF
import vim
num = vim.eval("a:number")
vim.command("normal gg{0}dd".format(num))
vim.command("w")
EOF
endfunction
可以調用上面函數Del(n)用來刪除當前緩沖區前n行的內容(只是示例而已,現實中別這么做!)上面用到了eval和command函數,如下:
vim模塊中兩個主要的方法
vim.command(str)
: 執行vim中的命令str(ex-mode,命令模式下的命令),返回值為None,比如:
:py vim.command("%s/aaa/bbb/g")
也可以用
vim.command("normal "+str)
來執行normal模式下的命令,比如說用以下命令刪除當前行的內容:
:py vim.command("normal "+'dd')
vim.eval(str)
: 用vim內部的解釋器來計算str中的內容,返回值可以是字符串、字典、或者列表,比如計算12+12的值:
:py print vim.eval("12+12")
將返回結算結果24。
前面的Del函數還提供了一個number參數,在vimL里面可以通過
let arg=a:number
來使用,在python中通過
vim.eval("a:number")
來使用。也可以通過參數位置來訪問,比如let arg=a:0或者是vim.eval("a:0")。我們可以使用"..."來代替命名參數來定義一個能接收任意數量參數的函數,不過這樣只能通過位置來訪問。
vim模塊還提供了一個
異常處理對象vim.error
,使用vim模塊時一旦出現錯誤,將會觸發一個vim.error異常,簡單的例子如下:
try:
??? vim.command("put a")
except vim.error:
??? # nothing in register a
vim模塊提供的對象
到這里你基本能用python來對緩沖區進行基本的操作,比如刪除行或者是在指定行添加內容等。不過在緩沖區添加內容會很不pythoner,因為你得使用command來調用vim的i/I/a/A命令。好在有更科學的方式,那就是利用vim模塊提供的對象來進行操作,看下面簡單的例子:
function! Append()
python << EOF
import vim
cur_buf = vim.current.buffer
lens = len(cur_buf)
cur_buf.append('" Demo', lens)
EOF
endfunction
Append函數在當前緩沖區的結尾添加注釋內容
" Demo
,緩沖區對象是怎么一會兒事呢?
緩沖區對象
vim模塊提供了緩沖區對象來讓我們對緩沖區進行操作,該對象有兩個只讀屬性name和number,name為當前緩沖區文件的名稱(包含絕對路徑),number為緩沖區的數量。還有一個bool屬性valid,用來標識相關緩沖區是否被擦除。
緩沖區對象有以下幾種方法:
b.append(str): 在當前行的下面插入新的行,內容為str;b.append(str, n): 在第n行的下面插入新的行,內容為str;b.append(list)
b.append(list, n): 插入多行到緩沖區中;b.range(s,e): 返回一個
range對象
表示緩沖區中s到e行的內容。
注意使用append添加新行str時,str中一定不能包含換行符"\n"。str結尾可以有"\n",但會被忽略掉。
緩沖區對象的range方法會返回一個range對象來代表部分的緩沖區內容,那么range對象又有那些屬性以及方法呢? 其實在操作上range對象和緩沖區對象基本相同,除了range對象的操作均是在指定的區域上。range對象有兩個屬性start和end,分別是range對象的起始和結尾行。它的方法有r.append(str),r.append(str, n)和r.append(list),r.append(list, n)。
我們可以通過
vim.windows
來獲取vim中的窗口對象,我們只能通過窗口對象的屬性來對其進行操作,因為它沒有提供方法或者其他接口來操作。其中只讀屬性有buffer、number、tabpage等,讀寫屬性有cursor、height、width、valid等。具體可以查看幫助
:h python-window
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
