Lua和C/C++互相调用实例分析

 更新时间:2019年01月17日 10:19:26   作者:修语讲编程  
今天小编就为大家分享一篇关于Lua和C/C++互相调用实例分析,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧

lua作为小巧精悍的脚本语言,易于嵌入c/c++中 , 广泛应用于游戏AI ,实际上在任何经常变化的逻辑上都可以使用lua实现,配合c/c++实现的底层接口服务,能够大大降低系统的维护成本。下面对lua和c/c++的交互调用做一个实例分析:

lua提供了API用于在c/c++中构造lua的运行环境,相关接口如下:

//创建lua运行上下文
lua_State*
luaL_newstate(void) ;
//加载lua脚本文件
int luaL_loadfile(lua_State *L, const
char *filename);

lua和c/c++的数据交互通过”栈”进行

,操作数据时,首先将数据拷贝到”栈”上,然后获取数据,栈中的每个数据通过索引值进行定位,索引值为正时表示相对于栈底的偏移索引,索引值为负时表示相对于栈顶的偏移索引,索引值以1或-1为起始值,因此栈顶索引值永远为-1,栈底索引值永远为1 。 “栈”相当于数据在lua和c/c++之间的中转地。每种数据都有相应的存取接口 。

数据入”栈”接口:

void (lua_pushnil) (lua_State *L);
void (lua_pushnumber) (lua_State *L, lua_Number n);
void (lua_pushinteger) (lua_State *L, lua_Integer n);
void (lua_pushlstring) (lua_State *L, const char *s, size_t l);
void (lua_pushstring) (lua_State *L, const char *s);
void (lua_pushboolean) (lua_State *L, int b);
void (lua_pushcclosure) (lua_State *L, lua_CFunction fn, int n);

数据获取接口:

lua_Number (lua_tonumber) (lua_State *L, int idx);
lua_Integer (lua_tointeger) (lua_State *L, int idx);
int (lua_toboolean) (lua_State *L, int idx);
const char *(lua_tolstring) (lua_State *L, int idx, size_t *len);
lua_CFunction (lua_tocfunction) (lua_State *L, int idx);

“栈”操作接口:

int (lua_gettop) (lua_State *L);
void (lua_settop) (lua_State *L, int idx);
void (lua_pushvalue) (lua_State *L, int idx);
void (lua_remove) (lua_State *L, int idx);
void (lua_insert) (lua_State *L, int idx);
void (lua_replace) (lua_State *L, int idx);
int (lua_checkstack) (lua_State *L, int sz);

lua中定义的变量和函数存放在一个全局table中,索引值为LUA_GLOBALSINDEX,table相关操作接口:

void (lua_gettable) (lua_State *L, int idx);
void (lua_getfield) (lua_State *L, int idx, const char *k);
void (lua_settable) (lua_State *L, int idx);
void (lua_setfield) (lua_State *L, int idx, const char *k);

当”栈”中包含执行脚本需要的所有要素(函数名和参数)后,调用lua_pcall执行脚本:

int (lua_pcall) (lua_State *L, int nargs, int nresults, int errfunc);

下面进行实例说明:

func.lua

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
//lua头文件
#include <lua.h>
#include <lualib.h>
#include <lauxlib.h>
#define err_exit(num,fmt,args) \
  do{printf("[%s:%d]"fmt"\n",__FILE__,__LINE__,##args);exit(num);} while(0)
#define err_return(num,fmt,args) \
  do{printf("[%s:%d]"fmt"\n",__FILE__,__LINE__,##args);return(num);} while(0)
//lua中调用的c函数定义,实现加法
int csum(lua_State* l)
{
  int a = lua_tointeger(l,1) ;
  int b = lua_tointeger(l,2) ;
  lua_pushinteger(l,a+b) ;
  return 1 ;
}
int main(int argc,char** argv)
{
  lua_State * l = luaL_newstate() ;    //创建lua运行环境
  if ( l == NULL ) err_return(-1,"luaL_newstat() failed");
  int ret = 0 ;
  ret = luaL_loadfile(l,"func.lua") ;   //加载lua脚本文件
  if ( ret != 0 ) err_return(-1,"luaL_loadfile failed") ;
  ret = lua_pcall(l,0,0,0) ;
  if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
  lua_getglobal(l,"width");       //获取lua中定义的变量
  lua_getglobal(l,"height");
  printf("height:%ld width:%ld\n",lua_tointeger(l,-1),lua_tointeger(l,-2)) ;
  lua_pop(l,1) ;            //恢复lua的栈
  int a = 11 ;
  int b = 12 ;
  lua_getglobal(l,"sum");        //调用lua中的函数sum
  lua_pushinteger(l,a) ;
  lua_pushinteger(l,b) ;
  ret = lua_pcall(l,2,1,0) ;
  if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
  printf("sum:%d + %d = %ld\n",a,b,lua_tointeger(l,-1)) ;
  lua_pop(l,1) ;
  const char str1[] = "hello" ;
  const char str2[] = "world" ;
  lua_getglobal(l,"mystrcat");     //调用lua中的函数mystrcat
  lua_pushstring(l,str1) ;
  lua_pushstring(l,str2) ;
  ret = lua_pcall(l,2,1,0) ;
  if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
  printf("mystrcat:%s%s = %s\n",str1,str2,lua_tostring(l,-1)) ;
  lua_pop(l,1) ;
  lua_pushcfunction(l,csum) ;     //注册在lua中使用的c函数
  lua_setglobal(l,"csum") ;      //绑定到lua中的名字csum
  lua_getglobal(l,"mysum");      //调用lua中的mysum函数,该函数调用本程序中定义的csum函数实现加法
  lua_pushinteger(l,a) ;
  lua_pushinteger(l,b) ;
  ret = lua_pcall(l,2,1,0) ;
  if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
  printf("mysum:%d + %d = %ld\n",a,b,lua_tointeger(l,-1)) ;
  lua_pop(l,1) ;
  lua_close(l) ;           //释放lua运行环境
  return 0 ;
}

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。如果你想了解更多相关内容请查看下面相关链接

相关文章

  • C++基础入门教程(一):基础知识大杂烩

    C++基础入门教程(一):基础知识大杂烩

    这篇文章主要介绍了C++基础入门教程(一):基础知识大杂烩,本文讲解了注释、头文件、命名空间等内容,需要的朋友可以参考下
    2014-11-11
  • 详解C语言中的Static关键字

    详解C语言中的Static关键字

    这篇文章主要为大家介绍了C语言中Static关键字,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-01-01
  • C语言动态内存分配的详解

    C语言动态内存分配的详解

    这篇文章主要介绍了C语言动态内存分配的详解的相关资料,这里提供了实现方法整理和出现错误的解决办法,需要的朋友可以参考下
    2017-07-07
  • C++实现简单的HTTP服务器

    C++实现简单的HTTP服务器

    这篇文章主要为大家详细介绍了C++实现简单的HTTP服务器的相关资料,感兴趣的朋友可以参考下
    2016-05-05
  • c语言实现单链表算法示例分享

    c语言实现单链表算法示例分享

    这篇文章主要介绍了c语言实现单链表算法示例,需要的朋友可以参考下
    2014-02-02
  • 浅谈C#中List<T>对象的深度拷贝问题

    浅谈C#中List<T>对象的深度拷贝问题

    下面小编就为大家带来一篇浅谈C#中List<T>对象的深度拷贝问题。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-01-01
  • Qt实现模糊匹配功能的实例详解

    Qt实现模糊匹配功能的实例详解

    对于浏览器的使用,我想大家一定不会陌生吧,输入要搜索的内容时,会出现相应的匹配信息。本文就来用Qt实现模糊匹配功能,感兴趣的可以了解一下
    2022-10-10
  • C语言实现快速排序的方法及优化

    C语言实现快速排序的方法及优化

    这篇文章主要介绍了C语言实现快速排序的方法及优化,快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法,下面我们来看一看传说中的快速排序的特点与效率怎么样,需要的朋友可以参考下
    2023-07-07
  • C++11 上下文关键字的具体实践

    C++11 上下文关键字的具体实践

    本文主要介绍了C++11 上下文关键字的具体实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-06-06
  • C++ RTTI与4种类型转换的深入理解

    C++ RTTI与4种类型转换的深入理解

    这篇文章主要给大家介绍了关于C++ RTTI与4种类型转换的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03

最新评论