Lua和C之间的交互

转自:http://blog.csdn.net/sumoyu/article/details/2592693

(一) Lua 调C函数

  1. 什么样类型的函数可以被Lua调用
 
typedef int (*lua_CFunction) (lua_State *L);
  1. 符合类型的函数怎样处理后才可以被Lua调用

使用lua_register或者 lua_pushfunction和lua_setglobal()把要调用的函数加入到lua状态机中。

#define lua_register(L,n,f) /
 (lua_pushcfunction(L, f), lua_setglobal(L, n))

lua_register的第二个参数就是Lua脚本中对这个函数的调用名称。

举例:

如果C函数名称是foo,使用lua_registe注册(L,”acfoo”,foo),那么在Lua脚本中使用acfoo来表示使用foo函数.

  1. Lua 如何调用c函数

简单,使用注册的名称直接调用

  1. 如何传递参数和计算结果

① 使用堆栈交互

引用使用手册上的一段话:

Lua 使用一个虚拟栈来和 C 传递值。栈上的的每个元素都是一个 Lua 值(nil,数字,字符串,等等)。

 

无论何时 Lua 调用 C,被调用的函数都得到一个新的栈,这个栈独立于 C 函数本身的堆栈,也独立于以前的栈。(在 C 函数里,用 Lua API 不能访问到 Lua 状态机中本次调用之外的堆栈中的数据),它里面包含了 Lua 传递给 C 函数的所有参数,而 C 函数则把要返回的结果也放入堆栈以返回给调用者。

 

方便起见,所有针对栈的 API 查询操作都不严格遵循栈的操作规则。而是可以用一个索引来指向栈上的任何元素:正的索引指的是栈上的绝对位置(从一开始);负的索引则指从栈顶开始的偏移量。更详细的说明一下,如果堆栈有 n 个元素,那么索引 1 表示第一个元素(也就是最先被压入堆栈的元素)而索引 n 则指最后一个元素;索引 -1 也是指最后一个元素(即栈顶的元素),索引 -n 是指第一个元素。如果索引在 1 到栈顶之间(也就是,1 ≤ abs(index) ≤ top)我们就说这是个有效的索引

② 从Lua脚本中获取参数

int n = lua_gettop(L);
/* get each argument */
lua_tostring(lua_State *L, int index)
     …….

index: 1—左边第一个参数,2—左边第二个参数,......

③ 返回返回值

按顺序返回,Lua按照返回顺序接受

Lua_pushXXX(L,第一个返回值)

Lua_pushXXX(L,第二个返回值)

………

Lua调用C函数例子:

C程序:

static int average(lua_State *L)

{

    /* get number of arguments */

    int n = lua_gettop(L);

    double sum = 0;

    int i;

    /* loop through each argument */

    for (i = 1; i <= n; i++)

    {

        if (!lua_isnumber(L, i))

        {

            lua_pushstring(L, "Incorrect argument to ‘average‘");

            lua_error(L);

        }

        /* total the arguments */

        sum += lua_tonumber(L, i);

    }

    /* push the average */

    lua_pushnumber(L, sum / n); //第一个返回值

    /* push the sum */

    lua_pushnumber(L, sum); //第二个返回值

    /* return the number of results */

    return 2;

}

void LuaCallC()

{

    /* initialize Lua */

    lua_State * L = lua_open();

    /* load Lua base libraries */

    luaL_openlibs(L);

    /* register our function */

    lua_register(L, "average", average);

    /* run the script */

    luaL_dofile(L, "average.lua");

    /* cleanup Lua */

    lua_close(L);

}

 

Lua脚本,average.lua:

avg, sum = average(20,40,50,60,80)

print("The average is ", avg)

print("The sum is ", sum)

(二)            Lua 从C库中调用

  1. 生成C函数库

① 所有可以被Lua调用的函数必须是lua_CFunction类型

 

②  所有被调用的函数加入到一个luaL_reg数组中

③ 一个luaopen_*(*表示库的名称)供lu调用库时打开库

使用luaL_register(lua_State *L,

const char *libname,

const luaL_Reg *l)

libname,注册lua使用这个库时的使用名称

luaL_Reg *l,把luaL_Reg数组里的函数注册到lua栈里,供lua调用

注意:BCB默认导出的c函数前面加了下划线,因此在动态库工程中加入一个def文件,在生成时不用加下划线。内容是:

   Export

       FunName = _FunName (FunName表示要导出的函数名称,Lua使用的库中就是luaopen_*)

  1. Lua使用c库

require(libname) – 打开使用的库

libname.FunName – 使用c库中提供的函数

Lua调用C函数库例子:

C库代码,C函数的名称”dllforlua.dll”

static int lua_msgbox(lua_State* L)

{

    const char* message = luaL_checkstring(L, 1);

    const char* caption = luaL_optstring(L, 2, "");

    int result = MessageBox(NULL, message, caption, MB_YESNO);

    lua_pushnumber(L, result);

    return 1;

}

static const  luaL_Reg mylib[] =

{

    {"msgbox", lua_msgbox},

    {NULL, NULL}

};

int __declspec(dllexport)  luaopen_dllforlua(lua_State* L)

{

    luaL_register(L, "dllforlua", mylib);

    return 1;

}

 

Lua脚本,Test.lua

require(“dllforlu”)

dllforlua.msgbox("Hey, it worked!", "Lua Message Box");

 

(三)            C调Lua 函数

  1. 初始化Lua环境

Lua_open或者:lua_newstate

luaL_newstate(调用lua_newstate,并且设置了一个恐慌函数)

  1. 加载Lua标准库

Lua_openlibs(打开所有标准库)

不打开所有库,打开需要的库:

Luaopen_base

luaopen_package

luaopen_string

luaopen_table

luaopen_math

……….

  1. 加载Lua和函数函数

luaL_dofile()

lua_getglobal()

大小写敏感,名字于Lua脚本的函数名称大小写完全一致

  1. 压入参数

不同类型采用不同的函数,按照从左往右的顺序依次压栈

lua_pushnumber,lua_pushstring,…..

  1. 执行函数

lua_call, lua_pcall

  1. 获取返回值

不同类型使用不同的函数,注意索引,获取前要检查类型

  1. 从栈中弹出返回值

lua_pop()

  1. 关闭Lua状态机

lua_close()

C程序掉用Lua函数例子:

C函数:

void CCallLua()

{

    // Create a LUA VMachine

    lua_State *L;

    //L = luaL_newstate();

    L = lua_open();

    //Load Libraries

    luaL_openlibs(L);

     // 运行脚本 /

    luaL_dofile(L, "clua.lua");

    lua_getglobal(L,"Sum");

    lua_pushnumber(L,2);//第一个参数

    lua_pushnumber(L,3);//第二个参数

    lua_pushnumber(L,4);//第三个参数

    lua_pcall(L,3,2,0);

    double sum=0,ave=0;

    if(lua_isnumber(L,1))

    {

        sum=lua_tonumber(L,1);

    }

    if(lua_isnumber(L,2))

    {

        ave=lua_tonumber(L,2);

    }

    lua_pop(L,2);

    cout<<"Sum ="<<sum

        <<"/nAve ="<<ave<<endl;

    // 清除Lua

    lua_close(L);

    getchar();

}

 

Lua脚本Clua.lua:

function Sum(...)

  local s=0

  local num=0     

  for k,v in pairs{...} do

    s = s + v

       num = k

  end

  return s,s/num

end
时间: 2024-11-10 01:29:19

Lua和C之间的交互的相关文章

Lua和C语言的交互——C API

Lua可作为扩展性语言(Lua可以作为程序库用来扩展应用的功能),同时也是个可扩展的语言(Lua程序中可以注册由其他语言实现的函数). C和Lua交互的部分称为C API.C API是一个C代码与Lua进行交互的函数集.他由以下部分组成:读写Lua全局变量的函数.调用Lua函数的函数.运行Lua代码片断的函数.注册C函数然后可以在Lua中被调用的函数,等等. API中有些函数为了方便以宏的方式实现. 当在Lua和C之间交换数据时我们面临着两个问题:动态与静态类型系统的不匹配和自动与手动内存管理的

AngularJs-指令和指令之间的交互(动感超人)

前言: 上节我们学习到了指令和控制器之间的交互,通过给指令添加动作,调用了控制器中的方法.本节我们学习指令和指令之间是如何交互的,我们通过一个小游戏来和大家一起学习,听大漠老师说这是国外的人写的demo,我们可以借鉴学习. 1,动感超人 上面的三个按钮,代表三个超人,在此想问下,哪些想看超人的朋友们是不是有种被骗了的感觉? 当我们的鼠标移动到哪个超人的身上的时候,就会输入这个超人所拥有的超能力(力量 + 敏捷 + 发光) <!DOCTYPE html> <html ng-app=&quo

Fragment与Activiy之间的交互

为了重用Fragment UI组件,我们应该把每一个fragment都构建成完全的自包含的.模块化的组件,定义他们自己的布局与行为.定义好这些模块化的Fragment后,就可以让他们关联activity,使他们与application的逻辑结合起来,实现全局的复合的UI. 通常fragment之间可能会需要交互,比如基于用户事件改变fragment的内容.所有fragment之间的交互需要通过他们关联的activity,两个fragment之间不应该直接交互. 定义一个接口 为了让fragmen

cocos2d-x 中LUA和平台之间的函数调用理解

先看一张流程图如下: 第一步: 先把NDKHelper中的方法转成LUA中可以调用的,转得方法可参考quick中LUA的用法.这个类主要是中间桥梁的作用,它可以根据是什么平台调用IOSNDKHelper或者AndroidSNDKHelper,这些判断都是用C语言来写的.还有就是对一些回调函数的保存机制和平台要用LUA中一些方法的保存. 第二步: IOS平台需要处理的就是IOSNDKHelper,这个类主要就是接收和发送给NDKHelper数据的方法,还有一个就是加载IOS平台的BasePlatf

Android学习笔记(十六)——碎片之间进行交互(附源码)

碎片之间进行交互 点击下载源码 很多时候,一个活动中包含一个或者多个碎片,它们彼此协作,向用户展示一个一致的UI.在这种情况下,碎片之间能进行通信并交换数据十分重要. 1.使用上一篇中创建的同一个项目,在fragment.xml中添加TextView的标识id: android:id="@+id/lblFragment1" 2.在fragment2.xml中添加一个Button,用于与fragment1进行交互: <Button android:id="@+id/btn

Salesforce视图与控制器之间的交互

刚接触Salesforce,过程的确是比较艰难了,中文资料几乎没有,看英文资料学的效率却不高,不过看了一段时间的英文资料发现自己英语水平挺高不少啊,现在看都不用工具翻译,早知道就再次尝试报个6级,看下能过不,嘻嘻....Salesforce的开发也是MVC模式,asp.net的MVC就玩的比较多了,换个平台一下子没适应过来,不过原理都一样,接下来就介绍一下最近的学习成果吧,来看一下SF中MVC模式下视图与控制器之间的交互,先贴控制器和视图的代码,下面有详细讲解. apex视图代码如下: <ape

深入分析:Android中app之间的交互

在我们开发Android App应用的时候,有些需求需要我们启动其他的App来处理一些逻辑,例如我们需要根据一个地址来调用系统或者相关的地图Map App,这样我们不用在自己的App中编写相应的功能,而是通过Intent来发送一些请求,调用相关的应用来处理这些请求.并且我们称这种Intent为隐式的Intent:这种隐式的Intent是相对于显式的Intent来讲的.显式的Intent我们都比较熟悉,显式的Intent常常需要声明类的名称,而隐式的Intent我们需要声明一个Action,我们A

深入分析:Android中app之间的交互(二,使用ComponentName)

在前一篇相关主题的博文中我们了解了如何使用Action来启动当前应用之外的Activity处理我们的业务逻辑,在本篇笔记中我在简单介绍一下使用ComponentName来与当前应用之外的应用进行交互. 在介绍Component之前,我们首先来了解ComponentName这个类:ComponentName与Intent同位于android.content包下,我们从Android官方文档中可以看到,这个类主要用来定义可见一个应用程序组件,例如:Activity,Service,Broadcast

对象之间的交互

之前写过一篇随笔<剪刀剪纸>是给一些新同事讲面向对象时用的,当时就感觉有些不顺畅,不过用来给新同事入门足够了就没多想,最近看书时偶尔走神把这件事想起来了,顺便群里讨论时谈到聚合之间的方法调用,于是决定写一篇博客纠正一下那篇随笔里的问题. 开头先声明一下,以下只是个例子,只是用来说明对象间交互的解耦,怎么样交互我觉得更好,但是如果是真的要写一个剪刀剪纸的程序,之前随笔的做法并不一定就是不好的,有些耦合只是在需要解的时候才应该去解.另外,以下做法只是理想的做法,但是现实的项目总会有各种各样的妥协,