用Golang運(yùn)行JavaScript的實(shí)現(xiàn)示例
C++太麻煩(難)了,想要盤弄一下V8實(shí)在是有些費(fèi)勁,但是Golang社區(qū)出了幾個(gè)Javascript引擎,要嘗試在別的語(yǔ)言中如何集成Javascript,是個(gè)不錯(cuò)的選擇。以下選了github.com/dop251/goja 來(lái)做例子。
Hello world
照著倉(cāng)庫(kù)的Readme,來(lái)一個(gè):
package main import ( "fmt" js "github.com/dop251/goja" ) func main() { vm := js.New() // 創(chuàng)建engine實(shí)例 r, _ := vm.RunString(` 1 + 1 `) // 執(zhí)行javascript代碼 v, _ : = r.Export().(int64) // 將執(zhí)行的結(jié)果轉(zhuǎn)換為Golang對(duì)應(yīng)的類型 fmt.Println(r) }
這個(gè)例子展示了最基本的能力,給定一段Javascript的代碼文本,它能執(zhí)行得到一個(gè)結(jié)果,并且能得到執(zhí)行結(jié)果的宿主語(yǔ)言的表示形式。
交互
Javascript和Golang之間的交互分成兩個(gè)方面:Golang向Javascript引擎中注入一些上下文,例如注冊(cè)一些全局函數(shù)供Javascript使用,創(chuàng)建一個(gè)對(duì)象等;Golang從Javascript引擎中讀取一些上下文,例如一個(gè)計(jì)算過(guò)程的計(jì)算結(jié)果。先看第一類。
常用的手段是,通過(guò)Runtime類型提供的Set方法在全局注冊(cè)一個(gè)變量,例如
... rts := js.New() rts.Set("x", 2) rts.RunString(`x+x`) // 4 ...
此處Set的方法簽名是func (r *Runtime) Set(name string, value interface{}),對(duì)于基本類型,不需要額外的包裝,就可以自動(dòng)轉(zhuǎn)換,但是當(dāng)需要傳遞一個(gè)復(fù)雜對(duì)象時(shí),需要用NewObject包裝一下:
rts := js.New() o := rts.NewObject() o.Set("x", 2) rts.Set("o", o) rts.RunString(`o.x+o.x`) // 4
切換到Golang的視角,是個(gè)很自然的過(guò)程,想要?jiǎng)?chuàng)建一個(gè)對(duì)象,需要在Golang中先創(chuàng)建一個(gè)對(duì)應(yīng)的表述,然后在Javascript中才能使用。對(duì)于更復(fù)雜的對(duì)象,嵌套就好了。
定義函數(shù)則有所不同,不同之處在于Javascript中的函數(shù)在Golang中的表示和其它類型的值不太一樣,Golang中表式Javascript中的函數(shù)的簽名為:func (js.FunctionCall) js.Value,js.FunctionCall中包含了調(diào)用函數(shù)的上下文信息,基于此我們可以嘗試給Javascript增加一個(gè)console.log的能力:
... func log(call js.FunctionCall) js.Value { str := call.Argument(0) fmt.Print(str.String()) return str } ... rts := js.New() console := rts.NewObject() console.Set("log", log) rts.Set("console", console) rts.RunString(`console.log('hello world')`) // hello world
相較于向Javascript引擎中注入一些信息,從中讀取信息則比較簡(jiǎn)單,前面的hello world中展示了一種方法,執(zhí)行一段Javascript代碼,然后得到一個(gè)結(jié)果。但是這種方法不夠靈活,如果想要精確的得到某個(gè)上下文,變量的值,就不那么方便。為此,goja提供了Get方法,Runtime類型的Get方法可以從Runtime中讀取某個(gè)變量的信息,Object類型的Get方法則可以從對(duì)象中讀取某個(gè)字段的值。簽名如下:func (r *Runtime) Get(name string) Value,func (o *Object) Get(name string) Value。但是得到的值的類型都是Value類型,想要轉(zhuǎn)換成對(duì)應(yīng)的類型,需要通過(guò)一些方法來(lái)轉(zhuǎn)換,這里就不再贅述,有興趣可以去看它的文檔。
一個(gè)復(fù)雜些的例子
goja值提供了基本的解析執(zhí)行Javascript代碼的能力,但是我們常見的宿主提供的能力,需要在使用的過(guò)程中自己去補(bǔ)充。下面就基于上面的技巧,提供一個(gè)簡(jiǎn)單的require加載本地Javascript代碼的能力。
通過(guò)require加載一段Commjs格式Javascript代碼,直觀的流程:根據(jù)文件名,讀取文本,組裝成一個(gè)立即執(zhí)行函數(shù),執(zhí)行,然后返回module對(duì)象,但是中間可以做一些小優(yōu)化,比如已經(jīng)被加載過(guò)的代碼, 就不重新加載,執(zhí)行,只是返回就好了。大概的實(shí)現(xiàn)如下:
package core import ( "io/ioutil" "path/filepath" js "github.com/dop251/goja" ) func moduleTemplate(c string) string { return "(function(module, exports) {" + c + "\n})" } func createModule(c *Core) *js.Object { r := c.GetRts() m := r.NewObject() e := r.NewObject() m.Set("exports", e) return m } func compileModule(p string) *js.Program { code, _ := ioutil.ReadFile(p) text := moduleTemplate(string(code)) prg, _ := js.Compile(p, text, false) return prg } func loadModule(c *Core, p string) js.Value { p = filepath.Clean(p) pkg := c.Pkg[p] if pkg != nil { return pkg } prg := compileModule(p) r := c.GetRts() f, _ := r.RunProgram(prg) g, _ := js.AssertFunction(f) m := createModule(c) jsExports := m.Get("exports") g(jsExports, m, jsExports) return m.Get("exports") } 要想讓引擎能使用這個(gè)能力,就需要將require這個(gè)函數(shù)注冊(cè)到Runtime中, // RegisterLoader register a simple commonjs style loader to runtime func RegisterLoader(c *Core) { r := c.GetRts() r.Set("require", func(call js.FunctionCall) js.Value { p := call.Argument(0).String() return loadModule(c, p) }) }
完整的例子有興趣可看github.com/81120/gode
寫在后面
之前一直分不清Javascript引擎和Javascript執(zhí)行環(huán)境的界限,通過(guò)這個(gè)例子,有了一個(gè)很具體的認(rèn)識(shí)。而且,對(duì)Node本身的結(jié)構(gòu)也有了一個(gè)更清楚的認(rèn)知。在一些場(chǎng)景下,需要將一些語(yǔ)言嵌入到另一個(gè)語(yǔ)言中實(shí)現(xiàn)一些更靈活的功能和解耦,例如nginx中的lua,游戲引擎中的lua,mongodb shell中的Javascipt,甚至nginx官方頭提供了一個(gè)閹割版本的Javascript實(shí)現(xiàn)作為配置的DSL。那么在這種需要嵌入DSL的場(chǎng)景下,嵌入一個(gè)成熟語(yǔ)言的執(zhí)行引擎比自己實(shí)現(xiàn)一個(gè)DSL要簡(jiǎn)單方便得多。而且,各種場(chǎng)景下,對(duì)語(yǔ)言本身的要求也不盡相同,例如邊緣計(jì)算場(chǎng)景,嵌入式下,可以用Javascript來(lái)開發(fā),但是是不是需要一個(gè)完整的V8呢?對(duì)環(huán)境和性能有特殊要求的場(chǎng)景下,限制DSL,提供必要的宿主語(yǔ)言擴(kuò)展也是個(gè)不錯(cuò)的思路吧。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
一款好用的移動(dòng)端滾動(dòng)插件BetterScroll
BetterScroll 是一款重點(diǎn)解決移動(dòng)端各種滾動(dòng)場(chǎng)景需求的開源插件,用于滾動(dòng)列表、選擇器、輪播圖、索引列表、開屏引導(dǎo)等應(yīng)用場(chǎng)景,感興趣的一起來(lái)了解一下2021-09-09JavaScript定時(shí)器設(shè)置、使用與倒計(jì)時(shí)案例詳解
這篇文章主要介紹了JavaScript定時(shí)器設(shè)置、使用與倒計(jì)時(shí)案例,詳細(xì)分析了javascript定時(shí)器的設(shè)置、取消、循環(huán)調(diào)用并附帶一個(gè)倒計(jì)時(shí)功能應(yīng)用案例,需要的朋友可以參考下2019-07-07JS點(diǎn)擊動(dòng)態(tài)添加標(biāo)簽、刪除指定標(biāo)簽的代碼
這篇文章主要介紹了JS點(diǎn)擊動(dòng)態(tài)添加標(biāo)簽、刪除指定標(biāo)簽的代碼,在文中給大家補(bǔ)充介紹了js 更加輪播圖圖片張數(shù)動(dòng)態(tài)生成小圓點(diǎn)的方法,需要的朋友參考下實(shí)現(xiàn)代碼2018-04-04javascript 限制輸入和粘貼(IE,firefox測(cè)試通過(guò))
javascript 限制輸入和粘貼 IE和火狐2.x、火狐3.x下測(cè)試通過(guò)2008-11-11javascript制作的簡(jiǎn)單注冊(cè)模塊表單驗(yàn)證
通常在我們的HTML頁(yè)面表單中有大量的數(shù)據(jù)驗(yàn)證工作, 免不了要寫很多驗(yàn)證表單的js代碼,這是一項(xiàng)非常繁瑣枯燥的工作。很多程序員也會(huì)經(jīng)常遺漏這項(xiàng)工作。所以寫了這一 段JavaScript代碼提供給大家使用。使用起來(lái)很簡(jiǎn)單,大家拿回去自由擴(kuò)展吧2015-04-04js實(shí)現(xiàn)隨機(jī)點(diǎn)名小功能
這篇文章主要為大家詳細(xì)介紹了js實(shí)現(xiàn)隨機(jī)點(diǎn)名小功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-0810 種最常見的 Javascript 錯(cuò)誤(頻率最高)
本文是小編給大家收藏的JavaScript 中頻度最高的 10 種錯(cuò)誤,我們會(huì)告訴你什么原因?qū)е铝诉@些錯(cuò)誤,以及如何防止這些錯(cuò)誤發(fā)生。需要的朋友參考下2018-02-02