go 基础 WebAssembly

th.jpeg

Web开发中为什么需要 WebAssembly ,以及在实际开发中如何使用 WebAssembly?带着这些问题开始今天分享。


question-mark1.jpg

在进入正题前我们简单地回顾一下 web 发展的历史

  • 第一个web 网页在 1991 当时只是提供一些可以跳转的静态页
  • 随着 10 天就设计出来的 javascript 的出现,出现 web 应用 gmail
  • 随后就是 javascript 库盛行的时代
  1. 使用 jquery 操作 dom 的轻便灵巧,几乎让我们忘记了 javascript 原生是如何操作 DOM
  2. angularjs 带来第一个真正意义上的 SPA 的框架
  3. three.js 让用户在 web 端可以体验到 3D 效果
  • 然后就是现代 web 时代V8WegGLHtml5、serviceworks
    有了这些才让 google docs trello 和 mirosoft365 这些原来的桌面应用成为了 web 端的应用。

不过这些还不够,没有用根本上解决在浏览器端对图形处理以及渲染还有大型计算的天生不足的问题。

那么首先看一看什么是WebAssembly

  • 用于 web 的全新的底层的字节码
  • 字节码是可以由其他语言(相对于 javascript 来说的其他语言)编译得来
  • 可以提供 web 的性能

现在不仅是理论上,会给大家分享一些 webassembly 的实现,并且会展望一些未来
首先我们来看一看 WebAssembly 究竟给我们带来了什么,以及选择 WebAssembly 的理由。来回答第一个问题

  • 当然首先选择了 WebAssembly 的目的就是为了提高 web 的性能,运行在浏览器端字节码会更快这点毋庸置疑。所以 WebAssembly 所能够提供的性能是无法通过 javascript 能够实现的。虽然 javascript 的引擎已经尽可能地提高了 javascript 的性能。但是 WebAssembly 在大型计算、图形处理上还是有距离的。
  • 还有就是 WebAssembly 的友好性,可以将其他类似 c++ 的语言编译为 WebAssembly 来使用,也可以使用开源的第三方。我们知道 c++ 可以 Android 和 ios 平台编写,同样也支持 Web 端
  • 给我们编写 Web 应用除了 javascript 以外提供更多语言的选择(c++ go rust 这是我知道的可能更多)以后 WebAssembly 还将会支持 kotlin 和 .Net
1523691204_go-webassembly.png

现在我们这里用 go 语言来给大家演示我们如何使用 WebAssembly 来回答第二个问题。
今天我带大家用 go 语言来实现一个 hello world 的 demo。go 语言对于 web 开发者总是那么友好。


600_468861629.jpeg.png

准备工作

需要做一些准备工作安装 gopherjs ,gopherjs 可以将 go 编译为 javascript 后就可以运行在浏览器上。

go get -u github.com/gopherjs/gopherjs

我这里使用 gopm 这个工具下载安装gopherjs

gopm get -v -g  github.com/gopherjs/gopherjs

搭建项目

cp $(go env GOROOT)/misc/wasm/wasm_exec.{html,js} .

通过上面的命令可以初始化以下两个文件为:

  • wasm_exec.js
  • wasm_exec.html
WebAssembly.instantiateStreaming(fetch("test.wasm"), go.importObject)

wasm_exec.html文件中,我们看到调用assembly文件代,所以我们需要将 main.go 文件编译为test.wasm文件,这样在 js 中才可以访问到 WebAssembly 所提供的方法。
下面为完整的 wasm_exec.html,可以将 html 名修改为 index.html 便于访问。

<!doctype html>
<!--
Copyright 2018 The Go Authors. All rights reserved.
Use of this source code is governed by a BSD-style
license that can be found in the LICENSE file.
-->
<html>

<head>
    <meta charset="utf-8">
    <title>Go wasm</title>
</head>

<body>
    <script src="wasm_exec.js"></script>
    <script>
        if (!WebAssembly.instantiateStreaming) { // polyfill
            WebAssembly.instantiateStreaming = async (resp, importObject) => {
                const source = await (await resp).arrayBuffer();
                return await WebAssembly.instantiate(source, importObject);
            };
        }

        const go = new Go();
        let mod, inst;
        WebAssembly.instantiateStreaming(fetch("test.wasm"), go.importObject).then((result) => {
            mod = result.module;
            inst = result.instance;
            document.getElementById("runButton").disabled = false;
        });

        async function run() {
            console.clear();
            await go.run(inst);
            inst = await WebAssembly.instantiate(mod, go.importObject); // reset instance
        }
    </script>

    <button onClick="run();" id="runButton" disabled>Run</button>
</body>

</html>
if (!WebAssembly.instantiateStreaming) { // polyfill
            WebAssembly.instantiateStreaming = async (resp, importObject) => {
                const source = await (await resp).arrayBuffer();
                return await WebAssembly.instantiate(source, importObject);
            };
        }

首先需要判断WebAssembly.instantiateStreaming是否存在,如果不存在我们通过其他方法实现WebAssembly.instantiateStreaming函数功能。
WebAssembly.instantiateStreaming异步加载以 *.wasm后缀结束的 webAssembly 的文件,成功加载test.wasm文件后取消 run 按钮的禁用,run 按钮的点击事件是运行

WebAssembly.instantiateStreaming(fetch("test.wasm"), 

这里可以写简单的 hello world ,用 go run main.go 命令查看一下输出。

package main

func main() {
    println("Hello World")
}

然后通过命令下面的命令来将main.go编译为test.wasm文件

GOARCH=wasm GOOS=js go build -o test.wasm main.go

接下来我们还需要写一个server.go来启动我们服务运行 wasm_exec.html

package main
import(
    "flag"
    "log"
    "net/http"
    // "strings"
)

var (
    listen = flag.String("listen",":8080","listen address")
    dir = flag.String("dir",".","directory to serve")
)

func main() {
    flag.Parse()
    log.Printf("listening on %q...",*listen)
    log.Fatal(http.ListenAndServe(*listen, http.FileServer(http.Dir(*dir))))
    // log.Fatal(http.ListenAndServe(*listen, http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request){
    //  if strings.HasSuffix(req.URL.Path,".wasm"){
    //      resp.Header().Set("content-type","application/wasm")
    //  }
    //  http.FileServer(http.Dir(*dir)).ServeHTTP(resp, req)
    // })))
}

在地址栏中输入localhost:8080/index.html"看到我们如下图,这里有一个 run 按钮点击就可以调用我们的 test.wasm 中输出方法

屏幕快照 2019-04-27 下午1.17.20.png

这样在 main.go 中输出的hello world在浏览器中就成功输出了。

屏幕快照 2019-04-27 下午1.17.27.png

接下来让 webAssembly 提供一些计算函数供 web 调用,创建add函数接收两个参数,进行加法运算。然后在js.Global()提供的 Set()方法以回调方式将 WebAssembly 提供的 add 方法挂在 javascript 的全局对象的add属性上,这样在 chrome 的 console 中输入 add 就可以调用 webAssembly 的提供 add 方法。

package main

import(
    "syscall/js"
)

func add(i []js.Value){
    js.Global().Set("output",js.ValueOf(i[0].Int() + i[1].Int()))
    println(js.ValueOf(i[0].Int() + i[1].Int()).String())
}

func registerCallbacks(){
    js.Global().Set("add",js.NewCallback(add))
}

func main() {
    c := make(chan struct{},0)
        println("Go WebAssembly Initialized")
    registerCallbacks()
    <-c
}
  • 在 registerCallbacks 函数中,js.Global()获取 javascript 全局对象然后,通过 Set 方法 add 函数以回调的方式挂接到 javascript 的全局的 add 属性上。这里是通过 js.NewCallback(add)实现的。

  • c := make(chan struct{},0) 创建 channel,然后在<-c 让 goroutine 阻塞,以便代码被执行到

屏幕快照 2019-04-27 下午2.15.49.png
func subtract(i []js.Value){
    js.Global().Set("output",js.ValueOf(i[0].Int() - i[1].Int()))
    println(js.ValueOf(i[0].Int() - i[1].Int()).String())   
}

func registerCallbacks(){
    js.Global().Set("add",js.NewCallback(add))
    js.Global().Set("subtract",js.NewCallback(subtract))
}
屏幕快照 2019-04-27 下午3.24.49.png
        if (!WebAssembly.instantiateStreaming) { // polyfill
            WebAssembly.instantiateStreaming = async (resp, importObject) => {
                const source = await (await resp).arrayBuffer();
                return await WebAssembly.instantiate(source, importObject);
            };
        }

        const go = new Go();
        let mod, inst;
        WebAssembly.instantiateStreaming(fetch("test.wasm"), go.importObject).then((result) => {
            mod = result.module;
            inst = result.instance;
            document.getElementById("runButton").disabled = false;
        });

        async function run() {
            console.clear();
            await go.run(inst);
            inst = await WebAssembly.instantiate(mod, go.importObject); // reset instance
        }

修改下无需点击run我们将 test.wasm 进行加载。

        const go = new Go();
        let mod, inst;
        WebAssembly.instantiateStreaming(fetch("test.wasm"), go.importObject).then(async (result) => {
            mod = result.module;
            inst = result.instance;
            document.getElementById("runButton").disabled = false;
            await go.run(inst);
        });

        // async function run() {
        //  console.clear();

        //  inst = await WebAssembly.instantiate(mod, go.importObject); // reset instance
        // }
屏幕快照 2019-04-27 下午3.33.51.png
屏幕快照 2019-04-27 下午3.33.56.png

通过 gopherjs实现获取 dom 元素的值,因为类型是字符,通过strconv.Atoi方法将字符串转为int 型进行计算。

func add(i []js.Value){

    value1 := js.Global().Get("document").Call("getElementById",i[0].String()).Get("value").String()
    value2 := js.Global().Get("document").Call("getElementById",i[1].String()).Get("value").String()

    int1, _ := strconv.Atoi(value1)
    int2, _ := strconv.Atoi(value2)

    js.Global().Set("output",int1 + int2)
    println(int1 + int2)
}
    <input type="text" id="value1" />
    <input type="text" id="value2" />
    <button onClick="add('value1','value2');" id="runButton">Add</button>
    <button onClick="subtract(3,2);" id="runButton">Subtract</button>
屏幕快照 2019-04-27 下午3.44.24.png

这样做还不够我们还需要将计算的结果输出到 input 中,所以继续对程序改造。这里并不对 gopherjs 进行过多解释,大家可能对这些代码有些陌生,不过稍微熟悉 web 开发,这个应该不难理解一看就懂。

func add(i []js.Value){

    value1 := js.Global().Get("document").Call("getElementById",i[0].String()).Get("value").String()
    value2 := js.Global().Get("document").Call("getElementById",i[1].String()).Get("value").String()

    int1, _ := strconv.Atoi(value1)
    int2, _ := strconv.Atoi(value2)

    js.Global().Get("document").Call("getElementById",i[2].String()).Set("value",int1 + int2)
    // js.Global().Set("output",int1 + int2)
    // println(int1 + int2)
}

对应修改一下 onclick 方法的参数 onClick="add('value1','value2','result');"

    <input type="text" id="value1" />
    <input type="text" id="value2" />
    <button onClick="add('value1','value2','result');" id="runButton">Add</button>
    <button onClick="subtract(3,2);" id="runButton">Subtract</button>
    <input id="result" type="text" />
屏幕快照 2019-04-27 下午3.48.59.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354

推荐阅读更多精彩内容