本文主要记录下在日常开发过程中, 使用 github.com/yuin/gopher-lua 过程中需要注意的地方。

后续遇到其他的需要注意的事项再补充。

1、加载LUA_PATH环境变量

在实际开发中,我们会将一些公共的、可重复使用的代码封装起来,假如我们只是一些简单的处理,全部写在一个文件是没有问题的,维护起来也并不是很麻烦。但是当我们的需求变得复杂起来,或者需求调整的时候,我们还是将所有功能都写在同一个文件的时候,就变得不合理起来,后期的维护更是灾难性的,这个时候就需要将一些公共的、重复使用的代码,抽离出来,根据功能的不同分类,当做一个个模块,在使用的时候使用 require 导入进来,这样才合理。

既然我们需要将自定义的 module 导入进来,那么我们肯定是需要设计环境变量的。

最重要的地方就是下面这点:

// 重点就是这句, 先设置环境变量, 再创建 lua 的 虚拟机, 这样环境变量才生效!!!
setLuaLibPath()
lvm := newLuaVm(pluginPath, context.Background())

接下来再详细看演示的demo。

先看看代码目录分层,有一个整体认识。

--go_call_lua_test
|---go.mod
|---main.go
|---plugin.lua
|---lib/lua/test.lua

main.go 代码

package main

import (
"context"
"errors"
"fmt"
lua "github.com/yuin/gopher-lua"
"go.uber.org/atomic"
luar "layeh.com/gopher-luar"
"os"
"path/filepath"
) func getExeDir() (string, string) {
exePath, _ := os.Executable()
exeDir, exeName := filepath.Split(exePath)
return exeDir, exeName
} func absPath(fp string) string {
exeAbsDir, _ := getExeDir()
//绝对路径
if !filepath.IsAbs(fp) {
return filepath.Join(exeAbsDir, fp)
} else {
return fp
}
} func setLuaLibPath() {
//设置lua环境变量
pathStr := ""
LuaLibPath := []string{"./lib/lua/?.lua"}
for _, path := range LuaLibPath {
pathStr += ";" + absPath(path)
}
pathStr += ";;"
os.Setenv("LUA_PATH", pathStr)
} type LuaVm struct {
*lua.LState
path string
Loaded atomic.Bool
} func newLuaVm(path string, ctx context.Context) *LuaVm {
lv := &LuaVm{
path: path,
LState: lua.NewState(),
}
lv.SetContext(ctx)
if err := lv.load(); err != nil {
panic(fmt.Errorf("lvm error=%s", err))
}
return lv
} func (lv *LuaVm) load() error {
//加载工具 下面是将 go 提供的功能当做模块 给 lua 使用
//lv.SetGlobal("FILE", luar.New(lv.LState, &test.FileTool{})) if len(lv.path) == 0 {
return fmt.Errorf("plugin file empty")
}
if err := lv.DoFile(lv.path); err != nil {
return err
} lv.Loaded.Store(true)
return nil
} func (lv *LuaVm) CallLua() (result string, err error) { if err := lv.CallByParam(lua.P{
Fn: lv.GetGlobal("PluginTest"),
NRet: 2,
Protect: true,
}, luar.New(lv.LState, "I'm coming go!")); err != nil {
fmt.Printf("PrimaryIn error=%s", err)
}
retCode, _ := lv.Get(-1).(lua.LNumber)
lv.Pop(1) retPkt, _ := lv.Get(-1).(lua.LString)
lv.Pop(1) if retCode != 0 {
return "", errors.New("untreated")
} return string(retPkt), nil
} func main() {
pluginPath := "./plugin.lua" // 重点就是这句, 先设置环境变量, 再创建 lua 的 虚拟机, 这样环境变量才生效!!!
setLuaLibPath()
lvm := newLuaVm(pluginPath, context.Background()) ret, err := lvm.CallLua()
fmt.Println("ret --> ", ret)
fmt.Println("err --> ", err)
}

plugin.lua 代码

local TestLib = require("test")

function PluginTest()
print("coming PluginTest")
TestLib.CallTestFunc() return "I'm coming lua!"
end

test.lua 代码

justTest = {}

function justTest.CallTestFunc()
print("coming justTest.CallTestFunc")
end -- 一定要 return, 否则不生效
return justTest

编译后执行结果:

coming PluginTest
coming justTest.CallTestFunc
ret --> I'm coming lua!
err --> <nil>

github.com/yuin/gopher-lua 踩坑日记的更多相关文章

  1. AI相关 TensorFlow -卷积神经网络 踩坑日记之一

    上次写完粗浅的BP算法 介绍 本来应该继续把 卷积神经网络算法写一下的 但是最近一直在踩 TensorFlow的坑.所以就先跳过算法介绍直接来应用场景,原谅我吧. TensorFlow 介绍 TF是g ...

  2. 人工智能(AI)库TensorFlow 踩坑日记之一

    上次写完粗浅的BP算法 介绍 本来应该继续把 卷积神经网络算法写一下的 但是最近一直在踩 TensorFlow的坑.所以就先跳过算法介绍直接来应用场景,原谅我吧. TensorFlow 介绍 TF是g ...

  3. hexo博客谷歌百度收录踩坑日记

    title: hexo博客谷歌百度收录踩坑日记 toc: false date: 2018-04-17 00:09:38 百度收录文件验证 无论怎么把渲染关掉或者render_skip都说我的格式错误 ...

  4. Hexo搭建静态博客踩坑日记(二)

    前言 Hexo搭建静态博客踩坑日记(一), 我们说到利用Hexo快速搭建静态博客. 这节我们就来说一下主题的问题与主题的基本修改操作. 起步 chrome github hexo git node.j ...

  5. Hexo搭建静态博客踩坑日记(一)

    前言 博客折腾一次就好, 找一个适合自己的博客平台, 专注于内容进行提升. 方式一: 自己买服务器, 域名, 写前端, 后端(前后分离最折腾, 不分离还好一点)... 方式二: 利用Hexo, Hug ...

  6. React Native Android配置部署踩坑日记

    万事开头难 作为一只进入ECMAScript世界不久的菜鸟,已经被React Native的名气惊到了,开源一周数万星勾起了我浓烈的兴趣.新年新气象,来个HellWorld压压惊吧^_^(故意少打个' ...

  7. 人工智能(AI)库TensorFlow 踩坑日记之二

    上次 踩坑日志之一 遗留的问题终于解决了,所以作者(也就是我)终于有脸出来写第二篇了. 首先还是贴上 卷积算法的示例代码地址 :https://github.com/tensorflow/models ...

  8. JavaScript 新手的踩坑日记

    引语 在1995年5月,Eich 大神在10天内就写出了第一个脚本语言的版本,JavaScript 的第一个代号是 Mocha,Marc Andreesen 起的这个名字.由于商标问题以及很多产品已经 ...

  9. 一次从Github向Gitee迁库踩坑记录

    先上结论:gitee的lfs功能收费,直接从github迁移包含lfs管理的文件会出错,配置lfs.url即可解决问题 Q: 为什么要迁库呢? A: github访问太慢了,不然我也不想费这个功夫! ...

  10. React Native踩坑日记 —— tailwind-rn

    项目背景 在项目的初始阶段,我们需要建立自己的design system,我们spike了一些方案,tailwind-rn就是其中一种,如果有用到或者即将用到tailwind-rn的,可以进来看一看, ...

随机推荐

  1. 在 Transformers 中使用对比搜索生成可媲美人类水平的文本 🤗

    1. 引言 自然语言生成 (即文本生成) 是自然语言处理 (NLP) 的核心任务之一.本文将介绍神经网络文本生成领域当前最先进的解码方法 对比搜索 (Contrastive Search).提出该方法 ...

  2. 用好kafka,你不得不知的那些工具

    前言 工欲善其事,必先利其器.本文主要分享一下消息中间件kafka安装部署的过程,以及我平时在工作中针对kafka用的一些客户端工具和监控工具. kafka部署架构 一个kafka集群由多个kafka ...

  3. KO之间互相调用

    需求 假设有两个KO,命名为moduleA.KO,moduleB.KO,现在要实现在moduleB.KO中调用moduleA.KO中的函数. 实现 ModuleA实现 源码: #include < ...

  4. 2019年蓝桥杯C/C++大学B组省赛真题(特别数的和)

    题目描述: 小明对数位中含有2.0.1.9 的数字很感兴趣(不包括前导0) 在1到40中这样的数包括1.2.9.10 至32.39 和40,共28 个,他们的和是574. 请问,在1到n 中,所有这样 ...

  5. 自然语言处理 Paddle NLP - 预训练语言模型及应用

    什么是语言理解? 关于疫情的一段对话: 中国:我们这边快完了 欧洲:我们这边快完了 中国:我们好多了 欧洲:我们好多了 挑战: 语言的复杂性和多样性 多义/同义/歧义现象 灵活多变的表达形式 语言背后 ...

  6. 从JDK源码级深入剖析main方法的运行机制

    如果你是一名Java应用开发工程师,你应该对"public static void main(String[] args)"这段代码再熟悉不过了,然而你是否了解main方法是如何调 ...

  7. PostgreSQL 12 文档: SQL 语法

    SQL 命令   这部分包含PostgreSQL支持的SQL命令的参考信息.每条命令的标准符合和兼容的信息可以在相关的参考页中找到. 目录 ABORT - 中止当前事务 ALTER AGGREGATE ...

  8. python中引用自己封装的包飘红线处理办法

    1.安装 opencv-contrib-python  可解决引用自己包名提示 无法识别 2.取消unresolved referencesde 的勾勾

  9. 查询mysql数据库目前有哪些链接 具体ip及数量

    SELECT substring_index(host, ':',1) AS host_name,state,count(*) FROM information_schema.processlist ...

  10. 【Python】数据可视化利器PyCharts在测试工作中的应用

    PyCharts 简介 PyCharts 是一个基于 Python 的数据可视化库,它支持多种图表类型,如折线图.柱状图.饼图等.PyCharts 提供了简洁的 API,使得用户能够轻松地创建各种图表 ...