lua与C交互

什么样类型的函数可以被Lua调用

typedef int (*lua_CFunction) (lua_State *L);

符合类型的函数怎样处理后才可以被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函数.

Lua 如何调用c函数   简单,使用注册的名称直接调用

如何传递参数和计算结果

① 使用堆栈交互

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

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-16 09:43:15

lua与C交互的相关文章

Lua与C++交互初探之Lua调用C++

Lua与C++交互初探之Lua调用C++ 上一篇我们已经成功将Lua的运行环境搭建了起来,也成功在C++里调用了Lua函数.今天我来讲解一下如何在Lua里调用C++函数. Lua作为一个轻量级脚本语言,他只包含了一些必要的系统库函数,当有需要时还得自己去写.有一次我要做一个两数异或的操作发现函数库里居然没有异或运算.不得不非常苦逼的自己去写.后来接触Lua深了之后才知道将这种"缺陷"可以由C函数来弥补.但要做到这一点对于一个对C只知道if else的学生来说确实还是有不少难度. 在学习

lua与c++交互初探

lua与c++交互初探 自从学习了lua这个脚本语言之后,无时不想着将他与c/c++联系起来,看看他真正的威力.奈何水平有限,网上找的代码无论怎样都无法运行成功.我知道是我少了某一步但是又不知道到底少了什么,于是就在各大博客.网站.论坛不断的摸索测试.我也不知道花了多长时间.总之在今天测试成功了.我把我测试遇到的问题和解决过程贴出来供大家参考. 一.lua环境的搭建 建议去网上下载luaforwindow,这是一款跟众多window系统的软件一样,安装起来简单方便,一路点next就能搞定了.而且

Lua和C++交互总结(很详细)

一不小心又标题党了?不好意思哈- 翻看了以前的博客,发下废话太多了自己都看不下去啊,恩,以后尽量写得简洁一些. 本文不涉及lua语法学习,如果有需要,请移步:http://book.luaer.cn/ 一.lua堆栈 要理解lua和c++交互,首先要理解lua堆栈. 简单来说,Lua和C/c++语言通信的主要方法是一个无处不在的虚拟栈.栈的特点是先进后出. 在lua中,lua堆栈就是一个struct,堆栈索引的方式可是是正数也可以是负数,区别是:正数索引1永远表示栈底,负数索引-1永远表示栈顶.

Lua 与C交互 第二篇

使用静态链接的方式 #include <stdio.h> #include <stdlib.h> #include <string.h> #ifdef __cplusplus extern "C" { #endif #include "lua/include/lua.h" #include "lua/include/lauxlib.h" #include "lua/include/lualib.h&q

用好lua+unity,让性能飞起来——lua与c#交互篇

前言 在看了uwa之前发布的<Unity项目常见Lua解决方案性能比较>,决定动手写一篇关于lua+unity方案的性能优化文. 整合lua是目前最强大的unity热更新方案,毕竟这是唯一可以支持ios热更新的办法.然而作为一个重度ulua用户,我们踩过了很多的坑才将ulua上升到一个可以在项目中大规模使用的状态.事实上即使到现在lua+unity的方案仍不能轻易的说可以肆意使用,要用好,你需要知道很多. 因此,这篇文章是从一堆简单的优化建议里头,逐步挖掘出背后的原因.只有理解了原因,才能很清

Lua和C++交互详细总结_1_Lua堆栈

一.Lua堆栈 要理解Lua和C++交互,首先要理解Lua堆栈. 简单来说,Lua和C/C++语言通信的主要方法是一个无处不在的虚拟栈.栈的特点是先进后出. 在Lua中,Lua堆栈就是一个struct,堆栈索引的方式可是是正数也可以是负数,区别是:正数索引1永远表示栈底,负数索引-1永远表示栈顶.如图: lua的栈类似于以下的定义, 它是在创建lua_State的时候创建的: TValue stack[max_stack_len]  // 欲知内情可以查 lstate.c 的stack_init

Lua和C++交互详细总结

转自:http://cn.cocos2d-x.org/tutorial/show?id=1474 一.Lua堆栈 要理解Lua和C++交互,首先要理解Lua堆栈. 简单来说,Lua和C/C++语言通信的主要方法是一个无处不在的虚拟栈.栈的特点是先进后出. 在Lua中,Lua堆栈就是一个struct,堆栈索引的方式可是是正数也可以是负数,区别是:正数索引1永远表示栈底,负数索引-1永远表示栈顶.如图: 二.堆栈的操作 因为Lua与C/C++是通过栈来通信,Lua提供了C API对栈进行操作. 我们

用好Lua+Unity,让万金6.0平台搭建性能飞起来——Lua与C#交互篇

前言万金6.0平台搭建论坛:haozbbs.com Q1446595067 在看了uwa之前发布的<Unity项目常见Lua解决方案性能比较>,决定动手写一篇关于lua+unity方案的性能优化文.整合lua是目前最强大的unity热更新方案,毕竟这是唯一可以支持ios热更新的办法.然而作为一个重度ulua用户,我们踩过了很多的坑才将ulua上升到一个可以在项目中大规模使用的状态.事实上即使到现在lua+unity的方案仍不能轻易的说可以肆意使用,要用好,你需要知道很多.因此,这篇文章是从一堆

lua与oc交互

//变成.mm文件,用于c++与oc交互; //    在.mm文件中可以实现c++与oc的混编,从而达到c++与oc的交互 //    c++与java的交互->lua(quick)与java的交互 //    luaj.callstaticmethod(“AppConTroller”,”ShowImagePicker”,{listener = callBack}) //    luaj 以及luaoc 都是放在luaBridge

Lua和C++交互详细总结_5_总结

Lua和C++是通过一个虚拟栈来交互的. C++调用Lua实际上是:由C++先把数据放入栈中,由Lua去栈中取数据,然后返回数据对应的值到栈顶,再由栈顶返回C++. Lua调C++也一样:先编写自己的C模块,然后注册函数到Lua解释器中,然后由Lua去调用这个模块的函数.