本文主要记录下在日常开发过程中, 使用 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. select_related和prefetch_related函数

    在数据库存在外键的其情况下,使用select_related()和prefetch_related()很大程度上减少对数据库的请求次数以提高性能 在表中查找数据的时候,使用外键表related_nam ...

  2. 【熊子q的代码乐园】用python写个健康报备记录小系统

    目录 一.前言 二.技术实现 1.概述 2. 环境 3. 技术核心 4. 数据库 5. 源代码 三.最后 一.前言 coding的一路上,遇到过许多问题,也写过一些代码去解决,回头看看还是有点意思的, ...

  3. 【原创】浅谈EtherCAT主站EOE(上)-EOE网络

    这篇文章的标题虽然是关于EtherCAT EOE,但其实主要内容是关于整个EOE网络结构,属于计算机网络原理.而EtherCAT EoE只是简单介绍,并不是文章的重点.需要注意的是,我们的描述主要基于 ...

  4. 逍遥自在学C语言 | 揭开while循环的神秘面纱

    前言 循环是一种重要的控制结构,可以使程序重复执行一段代码,直到满足特定条件为止. 在C语言中,while和do-while是两种常用的循环结构,本文将详细介绍这两种循环的用法. 一.人物简介 第一位 ...

  5. drf之频率类源码

    1 频率类 写一个类,继承SimpleRateThrottle,重写get_cache_key,返回[ip,用户id]什么,就以什么做限制,编写类属性 scope = 字符串,在配置文件中配置 'DE ...

  6. Framework 中使用 Toolkit.Mvvm 的生成器功能

    .NET Standard是.NET APIs的正式规范,可在多个.NET实现中使用..NET Standard的动机是为了在.NET生态系统中建立更大的统一性..NET 5及更高版本采用了不同的方法 ...

  7. nordic——NCS下的DFU升级(基于NCS)

    一.简介 在NCS中有多种的DFU选择,强烈推荐使用MCUboot,当然如果你需要选择传统的nrf_DFU也是可以的,但是要用到官方修改的源文件. 关于mcuboot,原理性的东西在官网和官方博客中有 ...

  8. 01-面试必会-JAVA基础篇

    1. Final 有什么用? 展开查看 被 final 修饰的类不可以被继承 被 final 修饰的方法不可以被重写 被 final 修饰的变量不可以被改变, 被 final 修饰不可变的是变量的引用 ...

  9. JavaCV的摄像头实战之十二:性别检测

    欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 本文是<JavaCV的摄像头实战> ...

  10. 解密Prompt系列10. 思维链COT原理探究

    前一章思维链基础和进阶玩法我们介绍了如何写Chain-of-thought Prompt来激活生成逐步推理,并提高模型解决复杂问题的能力,这一章我们追本溯源,讨论下COT的哪些元素是提升模型表现的核心 ...