[Unity3D]Unity3D游戏开发Lua随着游戏的债券(于)

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

喜欢我的博客请记住我的名字:秦元培。我的博客地址是blog.csdn.net/qinyuanpei

转载请注明出处,本文作者:秦元培, 本文出处:http://blog.csdn.net/qinyuanpei/article/details/39910099

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

大家好,我是秦元培,欢迎大家关注我的博客。我的博客地址是blog.csdn.net/qinyuanpei。在前一篇文章《Unity3D游戏开发之Lua与游戏的不解

之缘(上)》中,博主带领大家初步探索了Lua语言与游戏开发领域之间的紧密联系,今天让我们来继续将Lua语言进行究竟吧!通过前面的学习,我们知道设计Lua语言的目的是为了将Lua嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。Lua语言本身没有像其他语言提供丰富的类库,因此Lua语言必须依赖于其他语言来完毕功能上的扩展(但是正是在功能上牺牲才换来了Lua精简而稳定的核心)。假设我们要深入了解Lua语言的话,就必须要了解Lua语言与其他语言的交互接口,由于这将是我们使用Lua语言的基础。那么。今天就让博主来带领大家一起学习Lua语言与其他语言的交互吧!

一、Lua堆栈

假设我们想要理解Lua语言与其他语言交互的实质,我们首先就要理解Lua堆栈。

简单来说。Lua语言之所以能和C/C++进行交互。主要是由于存在这样一个无处不在的虚拟栈。

栈的特点是先进后出,在Lua语言中,Lua堆栈是一种索引能够是正数或者负数的结构,并规定正数1永远表示栈底。负数-1永远表示栈顶。

换句话说呢,在不知道栈大小的情况下。我们能够通过索引-1取得栈底元素、通过索引1取得栈顶元素。以下呢,我们通过一个实例来加深我们对于这段话的理解:

#include <iostream>

extern "C" {
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
}

using namespace std;

int main()
{
    //创建Lua环境
    lua_State* L=lua_open();
    //打开Lua标准库,经常使用的标准库有luaopen_base、luaopen_package、luaopen_table、luaopen_io、
    //luaopen_os、luaopen_string、luaopen_math、luaopen_debug
    luaL_openlibs(L);
    //压入一个数字20
    lua_pushnumber(L,20);
    //压入一个数字15
    lua_pushnumber(L,15);
    //压入一个字符串Lua
    lua_pushstring(L,"Lua");
    //压入一个字符串C
    lua_pushstring(L,"C");
    //获取栈元素个数
    int n=lua_gettop(L);
    //遍历栈中每一个元素
    for(int i=1;i<=n;i++)
    {
        cout << lua_tostring(L ,i) << endl;
    }
    return 0;
}

在上面的这段代码中,我们能够能够看到我们首先创建了一个lua_State类型的变量L,我们能够将它理解成一个Lua运行环境的上下文(Context),这里我们在Lua堆栈中压入了四个元素:20、15、"Lua"、"C"然后将其输出,假设大家理解了Lua堆栈中的索引,那么终于输出的结果应该是:20、15、"Lua"、"C"。由于索引1始终指向栈底,最先入栈的元素会处于栈底。

因此当我们依照递增的索引顺序来输出栈中的元素的话,实际上是自下而上输出,这样我们就能得到这种结果了。

好了。假设这段代码没有什么问题的话。接下来我们来解说Lua为C/C++提供的接口,它们均被定义在lua.h文件里。Lua提供的C/C++接口大部分与栈操作有关,因此深入理解Lua堆栈是学习Lua语言的重点和难点。通过数据结构的知识,我们能够知道栈有出栈和入栈两种基本操作,Lua提供的C API中入栈能够通过push系列的方法来实现。例如以下图所看到的:

而出栈或者说查询的方法则能够通过to系列的方法来实现,例如以下图:

这两部分是学习Lua语言一定要去了解的内容,由于以后假设须要我们将Lua整合到其他项目中这些内容,这些东西能够说是原理性、核心性的东西。

好了,以下我们利用这里的API对一个演示样例代码进行改造,这里添加了对栈中元素类型的推断:

#include <iostream>

extern "C" {
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
}

using namespace std;

int main()
{
    //创建Lua环境
    lua_State* L=lua_open();
    //打开Lua标准库,经常使用的标准库有luaopen_base、luaopen_package、luaopen_table、luaopen_io、
    //luaopen_os、luaopen_string、luaopen_math、luaopen_debug
    luaL_openlibs(L);
    //压入一个数字20
    lua_pushnumber(L,20);
    //压入一个字符串15
    lua_pushnumber(L,15);
    //压入一个字符串Lua
    lua_pushstring(L,"Lua");
    //压入一个字符串C
    lua_pushstring(L,"C");
    //获取栈中元素个数
    int n=lua_gettop(L);
    //遍历栈中每一个元素
    for(int i=1;i<=n;i++)
    {
        //类型推断
        switch(lua_type(L,i))
       {
          case LUA_TSTRING:
            cout << "This value‘s type is string" << endl;
          break;
          case LUA_TNUMBER:
            cout << "This value‘s type is number" << endl;
          break;
        }
        //输出值
        cout << lua_tostring(L ,i) << endl;
    }

    //释放Lua
    lua_close(L);
}

二、Lua与C++交互

Lua与C++的交互从宿主语言的选择划分上能够分为C++调用Lua和Lua调用C++两中类型:

1、C++调用Lua

使用C++调用Lua时我们能够直接利用C++中的Lua环境来直接Lua脚本,比如我们在外部定义了一个lua脚本文件。我们如今须要使用C++来訪问这个脚本该怎么做呢?在这里我们能够使用luaL_loadfile()、luaL_dofile()这两个方法个方法来实现。其差别是前者仅载入脚本文件而后者会在载入的同一时候调用脚本文件。我们一起来看以下的代码:

#include <iostream>

using namespace std;

#include <iostream>

extern "C" {
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
}

using namespace std;

int main()
{
    //创建Lua环境
    lua_State* L=luaL_newstate();
    //打开Lua标准库,经常使用的标准库有luaopen_base、luaopen_package、luaopen_table、luaopen_io、
    //luaopen_os、luaopen_string、luaopen_math、luaopen_debug
    luaL_openlibs(L);

    //以下的代码能够用luaL_dofile()来取代
    //载入Lua脚本
    luaL_loadfile(L,"script.lua");
    //运行Lua脚本
    lua_pcall(L,0,0,0);

    //将变量arg1压入栈顶
    lua_getglobal(L,"arg1");
    //将变量arg2压入栈顶
    lua_getglobal(L,"arg2");

    //读取arg1、arg2的值
    int arg1=lua_tonumber(L,-1);
    int arg2=lua_tonumber(L,-2);

    //输出Lua脚本中的两个变量
    cout <<"arg1="<<arg1<<endl;
    cout <<"arg2="<<arg2<<endl;

    //将函数printf压入栈顶
    lua_getglobal(L,"printf");
    //调用printf()方法
    lua_pcall(L,0,0,0);

    //将函数sum压入栈顶
    lua_getglobal(L,"sum");
    //传入參数
    lua_pushinteger(L,15);
    lua_pushinteger(L,25);
    //调用printf()方法
    lua_pcall(L,2,1,0);//这里有2个參数、1个返回值
    //输出求和结果
    cout <<"sum="<<lua_tonumber(L,-1)<<endl;

    //将表table压入栈顶
    lua_getglobal(L,"table");
    //获取表
    lua_gettable(L,-1);
    //输出表中第一个元素
    cout <<"table.a="<<lua_tonumber(L,-2)<<endl;

}

在这段代码中我们调用了一个外部的文件script.lua。这是一个Lua脚本文件,在调试阶段,我们须要将其放置在和C++项目源文件同级的文件夹下,而在正式运行阶段,我们仅仅须要将其和终于的可运行文件放在同一个文件夹下就好了。以下是脚本代码:

--在Lua中定义两个变量
arg1=15
arg2=20

--在Lua中定义一个表
table=
{
    a=25,
    b=30
}

--在Lua中定义一个求和的方法
function sum(a,b)
  return a+b
end

--在Lua中定义一个输出的方法
function printf()
  print("This is a function declared in Lua")
end

我们注意到在脚本文件里我们定义了一些变量和方法,在C++代码中我们首先用lua_getglobal()方法来讲Lua脚本中的变量或函数压入栈顶,这样我们就能够使用相关的to系列方法去获取它们,由于每次运行lua_getglobal()都是在栈顶。由于我们使用索引值-1来获取栈顶的元素。C++能够调用Lua中的方法,第一步和普通的变量同样。是将Lua中定义的方法压入栈顶。由于仅仅有压入栈中。我们才干够使用这种方法,接下来,我们须要通过push系列的方法为栈中的方法传入參数,在完毕參数传入后,我们能够使用一个lua_pcall()的方法来运行栈中的方法,它有四个參数。第一个參数是Lua环境状态Lua_State。第二个參数是要传入的參数个数,第三个參数是要返回的值的数目。第四个參数一般默认为0。

由于Lua支持返回多个结果。因此,我们能够充分利用Lua的这一特点来返回多个值。

运行该方法后。其结果会被压入栈顶,所以我们能够索引值-1来获取函数的结果。假设函数有多个返回值。则依照函数中定义的return 顺序,依次入栈,索引值-1代表最后一个返回值。好了。这就是C++调用Lua的详细实现了。

2、Lua调用C++

首先我们在C++中定义一个方法,该方法必须以Lua_State作为參数,返回值类型为int,表示要返回的值的数目。

static int AverageAndSum(lua_State *L)
{
	//返回栈中元素的个数
	int n = lua_gettop(L);
	//存储各元素之和
	double sum = 0;
	for (int i = 1; i <= n; i++)
	{
	    //參数类型处理
		if (!lua_isnumber(L, i))
		{
		    //传入错误信息
			lua_pushstring(L, "Incorrect argument to ‘average‘");
			lua_error(L);
		}
		sum += lua_tonumber(L, i);
	}
	//传入平均值
	lua_pushnumber(L, sum / n);
	//传入和
	lua_pushnumber(L, sum);

	//返回值的个数,这里为2
	return 2;
}

接下来我们在C++中使用lua_register()方法完毕对该方法的注冊

  lua_register(L, "AverageAndSum", AverageAndSum);

这样我们就能够在Lua环境中使用这种方法啦,前提是定义必须在运行代码之前完毕。我们在Lua脚本文件下添加对该方法的调用:

--在Lua中调用C++中定义并且注冊的方法
average,sum=AverageAndSum(20,52,75,14)
print("Average=".average)
print("Sum=".sum)

假设我们须要在C++中查看该方法调用的结果。那么这个在C++中调用Lua是一样的。

好了,C++和Lua的交互终于讲完了。被这块的代码纠结了好几天,这下总算是搞明确了。当然这仅仅是对原理的一种学习和理解啦。假设希望更好的使用Lua调用C++,建议了解这几个项目:

LuaPlusLuaBind。这样相信大家对于C++中的方法如何在Lua中绑定会有更好的认识吧!

三、Lua与C#交互

既然我们已经知道了C++是如何和Lua完毕交互的,理论上我们能够通过编写dll的方式将前面完毕的工作继续在C#中运行,但是这样做我们须要花费大量时间在三种语言之间纠结,由于这样会添加调试的难度。之前有个做coco2dx的朋友抱怨要在C++、Javascript、Lua之间来回跑,我当时没认为有什么,由于我最困难的时候就是C#和Java项目混合的情形,如今我算是深有体会了啊。这算是报应吗?哈哈。好了。不说这个了,好在C#与Lua的交互目方面前已经有了较好的解决方式。在开源社区我们能够找到非常多的支持在C#中调用Lua的工具库,博主这里向大家推荐的是LuaInterface这个开源项目。这个开源项目我找到了两个地址:

1、https://github.com/Jakosa/LuaInterface

2、http://code.google.com/p/luainterface

博主个人感觉这应该是同一个项目,由于两个项目的源码是一样的,只是从Github上下载的项目在使用的时候会报错。预计是我电脑里的Lua版本号和它项目里所用的Lua的版本号不一致造成的吧。以下的这个项目是能够使用的,博主这里写了一个简单的演示样例:

//------------------------------------------------------------------------------
// <summary>
//     这是一个用以演示LuaInterface的简单程序,通过LuaInterface我们能够实如今C#与Lua的
//     的相互通信。Lua是一个轻巧而高效的语言。它能够和不论什么语言混合使用。Lua语言最初并非
//     为游戏开发而诞生,却是由于游戏开发而成名。

眼下。在世界上有大量的游戏使用了Lua作为它
//     的脚本语言。如图Unity使用了C#作为它的语言。Lua在游戏开发领域发挥着不可忽视的重要作
//     用。

使用LuaInterface的方法例如以下:
//     1.C#
//     注冊Lua中可调用方法:
//    mLua.RegisterFunction(Lua调用方法名, 类, 类.GetMethod(C#方法名));
//    注:C#不要用法级泛型。即 void Fun<T>(string str);,假设使用,系统自己主动判定T为第一个參数的类型。

//	   载入Lua代码
//	   mLua.DoString(Lua代码);
//    mLua.DoFile(Lua文件绝对路径);
//     调用Lua方法
//	   mLua.GetFunction(Lua方法).Call(參数);  注:此处參数不要传递dynamic类型的类。否则Lua中无法获取属性值
//     2.Lua
//     调用C#方法时须要先注冊注冊后依照Lua方法处理
// </summary>
//------------------------------------------------------------------------------
using System;
using LuaInterface;
namespace LuaExample
{
	public class LuaScript
	{
		//定义LuaFile属性以便于从外部调用一个Lua脚本
		private string mLuaFile;
		public string LuaFile {
			get {
				return mLuaFile;
			}
			set {
				mLuaFile = value;
			}
		}

		//Lua虚拟机
		private Lua mLua;

		//构造函数
		public LuaScript ()
		{
			//初始化Lua虚拟机
			mLua=new Lua();
			//注冊Printf方法
			mLua.RegisterFunction("Printf",this,this.GetType().GetMethod("Printf"));
		}

		//定义一个C#方法供Lua使用
		public void Printf(string str)
		{
			Console.WriteLine("This Method is Invoked by Lua:" + str);
		}

		//在C#中调用Lua方法
		public void DoFile()
		{
			if(mLuaFile!="")
				//运行Lua脚本中的代码
				mLua.DoFile(mLuaFile);
		}

		//在C#中调用Lau方法
		public void DoString()
		{
			//以字符串形式定义的Lua脚本
			string mFuncString="function Add(a,b) io.write(a+b) end";
			//在Lua中定义该方法
			mLua.DoString(mFuncString);
			//调用该方法
			mLua.GetFunction("Add").Call(4,8);
		}

		//在Lua中调用C#脚本
		public void Invoke()
		{
			//调用注冊的Printf方法
			mLua.GetFunction("Printf").Call("Hello Lua");
		}
	}
}

接下来我们编写一个主类来调用这个类:

using System;
using LuaInterface;

namespace LuaExample
{
	class MainClass
	{
		public static void Main (string[] args)
		{
			//实例化LuaSxript
			LuaScript mLua=new LuaScript();
			//设置LuaFile
			mLua.LuaFile="D:\\test.lua";
			//调用字符串中定义的Lua方法
			mLua.DoString();
			//为美观考虑添加一个空行
			Console.WriteLine();
			//运行Lua文件里定义的脚本
			mLua.DoFile();
			//调用C#中定义的方法
			mLua.Invoke();
		}
	}
}

好了。C#与Lua的交互攻克了,很多其他的内容期待着大家自行到该项目源码中去寻找。好了,先这样吧!

四、Lua与Java交互

和C#相似的一点是在Java中我们能够使用JNI来调用C++代码。因此理论上Lua和Java应该是能够通过JNI来交互的。这块博主眼下没有展开研究。这里仅仅给大家推荐以下工具库:

1、LuaJava

2、luaj

五、结语

好吧。好了。好几天的时间来研究Lua语言的API,总算感觉是收获多一点吧。由于C++方面研究的东西不是非常多,所以像编译C++项目、配置C++环境、引用C++库和头文件这些问题曾经都不大会,这次居然一下子都学会了,博主推荐大家使用CodeBlocks这个C/C++开发环境,它内置的gcc编译器我认为还不错啦,并且它跨平台啊,以后工作了说不定会在Linux和Mac下做开发,选择一个跨平台的编辑器或者是IDE,对于我们来说未尝不是一件好事啊。由于学习新东西总是要花一定成本的。好了,今天的内容就是这样啦,希望大家喜欢啊,嘻嘻。突然认为这篇文章好长啊。

每日箴言:别总由于迁就别人就委屈自己。这个世界没几个人值得你总弯腰。弯腰的时间久了。仅仅会让人习惯于你的低姿态。你的不重要。

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

喜欢我的博客请记住我的名字:秦元培。我的博客地址是blog.csdn.net/qinyuanpei

转载请注明出处,本文作者:秦元培, 本文出处:http://blog.csdn.net/qinyuanpei/article/details/39910099

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

版权声明:本文博主原创文章。博客,未经同意不得转载。转载请注明作者和出处,谢谢。

时间: 2024-10-10 18:09:35

[Unity3D]Unity3D游戏开发Lua随着游戏的债券(于)的相关文章

[Unity3D]Unity3D游戏开发Lua随着游戏的债券(在)

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- 喜欢我的博客请记住我的名字:秦元培,我的博客地址是blog.csdn.net/qinyuanpei. 转载请注明出处,本文作者:

Android游戏开发:物理游戏之重力系统开发--圆形自由落体Demo

本节为大家提供有关物理游戏的知识,讲解了一个简单的圆形自由落体Demo的编写.. Java代码 package com.himi; import java.util.Random; import java.util.Vector; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import a

Unity 2D游戏开发教程之游戏中精灵的跳跃状态

Unity 2D游戏开发教程之游戏中精灵的跳跃状态 精灵的跳跃状态 为了让游戏中的精灵有更大的活动范围,上一节为游戏场景添加了多个地面,于是精灵可以从高的地面移动到低的地面处,如图2-14所示.但是却无法从低的地面移动到高的地面,因为当前的游戏精灵只能左右移动,即left和right.为了解决这个问题,本节就来为精灵添加跳跃状态.   图2-14  精灵从一个地面移动到另一个地面 (1)如果要为精灵添加跳跃状态,即jump,就不得不再引入其它状态: q   landing:用于表示精灵接触到地面

Unity 2D游戏开发教程之游戏精灵的开火状态

Unity 2D游戏开发教程之游戏精灵的开火状态 精灵的开火状态 "开火"就是发射子弹的意思,在战争类型的电影或者电视剧中,主角们就爱这么说!本节打算为精灵添加发射子弹的能力.因为本游戏在后面会引入敌人,而精灵最好具备开火的能力,否则会被敌人轻易干掉!具体的实现方法是: (1)导入一个表示子弹的图片到Unity,本示例中选用的子弹图片,名为PlayerBullet,如图2-23所示. 图2-23  导入到游戏项目的表示子弹的图片 (2)拖动此图片到Scene视图,即可在当前的游戏场景中

最大的幻术-游戏开发-我的游戏构思-环境

游戏,具体而言是游戏软件.那么,通过软件来作为载体,通过软件实现内容.一款软件能不能称为游戏在于这款软件所提供的功能是不是游戏体验. 什么是游戏,我无法给出定义.但是,游戏有什么,我们已经有了自己的答案.那么,现在我们想设计一款游戏,他是什么呢?游戏是什么类型的,有什么玩法,要达成什么目的.这里,我们先要停止思考这款游戏如何去吸引人,我们思考的是游戏软件所构建的虚拟的世界. 游戏让玩家在虚拟世界去干一件又一件的事情.因此,我最主要的关注点是游戏的场景,而作为游戏的战斗场景,我们一般称之为关卡.

Unity3D游戏开发之当游戏开发遇上Excel

各位朋友,大家好,我是秦元培,欢迎大家关注我的博客,我的博客地址是http://blog.csdn.net/qinyuanpei. 今天我们的话题是Excel,作为常用的办公软件的Excel相信大家都不陌生啦.可是如果我们认为Excel只是办公软件的话,那么这就不只是天真而是Out了.事实上Excel和游戏开发有着密切的联系,不知道大家还记不记得那款利用Excel开发出来的三国杀,这可能是Excel第一次以游戏开发的身份出现在大家面前吧.我们知道在游戏开发领域有一种工作叫做策划,就像在软件开发领

【VC++游戏开发】智力游戏——鸡蛋里挑骨头(仿扫雷)

在我学习游戏开发的过程中,遇到的最大的麻烦就是不知道一个游戏的完整实现过程,代码倒是其次. 这里,总结一下我做过的游戏.主要是梳理整每一个步骤. 先看下终于的效果 第1步,准备素材图片 包含鸡蛋.骨头,还有数字以及骨头标识和砸鸡蛋用的小锤. 第2步,声明 Egg类 它仅仅包括2个成员变量 class CEgg { public: int bitmap; //位图 int num; //数目 }; 第2步,初始化界面 1.1 还是要在CMainFrame中设置窗体參数以及图标 1.2 和 五子棋游

HTML5游戏开发,剪刀石头布小游戏案例

剪刀石头布,非常可爱的小游戏,相信大家都非常的怀念这款小游戏,小时候也玩过很多次,陪伴着我的童年的成长,现在是不是还会玩一下,剪刀石头布游戏的规则我们都知道是:剪刀剪布,石头砸剪刀,布包石头.跟朋友.同学.兄弟姐妹有意见分歧?通过“剪刀石头布游戏”来一局吧,谁赢了听谁的.躲猫猫的时候,通过“剪刀石头布游戏”来一局吧,谁输了谁找.洗衣服.做饭.扫地等等什么的,通过“剪刀石头布游戏”来一局吧,谁输了谁做.这是我的处女座游戏,学校的时候跟着培训老师一步一步写出来的,今天在这里将这款游戏分享给伙伴们,可

【读书笔记《Android游戏编程之从零开始》】20.游戏开发基础(游戏数据存储)

对于数据的存储,Android 提供了4种保存方式. (1)SharedPreference 此方法适用于简单数据的保持,文如其名,属于配置性质的保存,不适合比较大的情况,默认存放在手机内存里 (2)FileInputStream/FileOutputStream 此方式比较适合游戏的保存和使用,流文件数据存储可以保持较大的数据,而且通过此方式不仅能把数据存储在手机内存中,也能将数据保存到手机额SDcard中. (3)SQLite 此方式也适合游戏的保存和使用,不仅可以保存较大的数据,而且可以将