狭义的游戏对象是指游戏世界中所能看到及可交互的对象,如玩家、怪物、物品等,我们这里也主要讨论这类对象在服务器上的组织及实现。
在大部分的MMOG中,游戏对象的类型都大同小异,主要有物品、生物、玩家等。比如在wow中,通过服务器发下来的GUID我们可以了解到,游戏中有9大类对象,包括物品(Item)、背包(Container)、生物(Unit)、玩家(Player)、游戏对象(GameObject)、动态对象(DynamicObject)、尸体(Corpse)等。
在mangos的实现中,对象使用类继承的方式,由Object基类定义游戏对象的公有接口及属性,包括GUID的生成及管理、构造及更新UpdateData数据的虚接口、设置及获取对象属性集的方法等。然后分出了两类派生对象,一是Item,另一是WorldObject。Item即物品对象,WorldObject顾名思义,为世界对象,即可添加到游戏世界场景中的对象,该对象类型定义了纯虚接口,也就是不可被实例化,主要是在Object对象的基础上又添加了坐标设置或获取的相关接口。
Item类型又派兵出了一类Bag对象,这是一种特殊的物品对象,其本身具有物品的所有属性及方法,但又可作为新的容器类型,并具有自己特有的属性和方法,所以实现上采用了派生。mangos在实现时对Bag的类型定义做了点小技巧,Item的类型为2,Bag的类型为6,这样在通过位的方式来表示类型时,Bag类型也就同时属于Item类型了。虽然只是很小的一个技巧,但在很多地方却带来了极大的便利。
从WorldObject派生出的类型就有好几种了,Unit、GameObject、DynamicObject和Corpse。Unit为所有生物类型的基类,同WorldObject一样,也不可被实例化。它定义了生物类型的公有属性,如种族、职业、性别、生命、魔法等,另外还提供了相关的一些操作接口。游戏中实际的生物对象类型为Creature,从Unit派生,另外还有一类派生对象Player为玩家对象。Player与Creature在实现上最大的区别是玩家的操作由客户端发来的消息驱动,而Creature的控制是由自己定义的AI对象来驱动,另外Player内部还包括了很多的逻辑系统实现。
另外还有两类特殊的Creature,Pet和Totem,其对象类型仍然还是生物类,只是实现上与会有些特殊的东西需要处理,所以在mangos中将其作为独立的派生类,只是实现上的一点处理。另外在GameObject中也实现有派生对象,最终的继承关系图比较简单,就不麻烦地去画图了。
从我所了解的早期游戏实现来看,大部分的游戏对象结构都是采用的类似这种方式。可能与早期对面向对象的理解有关,当面向对象的概念刚出来时,大家认为继承就是面向对象的全部,所以处处皆对象,处处皆继承。
类实现的是一种封装,虽然从云风那里出来的弃C++而转投C的声音可能会影响一部分人,但是,使用什么语言本身就是个人喜好及团队整体情况决定的。我们所要的也是最终的实现结果,至于中间的步骤,完全看个人。还是用云风的话说,这只是一种信仰问题,我依然采用我所熟悉的C++,下面的描述也是如此。
随着面向对象技术的深入,以及泛型等概念的相继提出,软件程序结构方面的趋势也有了很大改变。C++大师们常说的话中有一句是这样说的,尽是采用组合而不是继承。游戏对象的实现也有类似的转变,趋向于以组合的方式来实现游戏对象类型,也就是实现一个通用的entity类型,然后以脚本定义的方式组合出不同的实际游戏对象类型。
描述的有些抽象,具体实现下一篇来仔细探讨下。
在游戏编程精粹四有三篇文章讲到了实体以及实体管理的实现方案,其中一篇文章说到了实体管理系统的四大要素:定义实体怎样沟通的实体消息,实现一实体类代码和数据的实体代码,维护已经注册在案的实体类列表,和用来创建、管理、发送消息的实体管理器。
关于实体消息的内容之前讨论事件机制的时候做过一点说明,其实这也就是按接口调用和按消息驱动的区别,现在mangos的做法是完全的接口调用,所以引擎内部就没有任何的实体消息。实体代码实现和实体管理器是我们重点要讨论的内容。
另有一篇文章也提到了使用类继续的方式实现游戏对象的两大问题,一是它要求系统中的所有对象都必须从一个起点衍生而成,也就是说所有对象类在编译的时候已经确定,这可能是一个不受欢迎的限制,如果开发者决定添加新的对象类,则必须要对基类有所了解,方能支持新类。另一个问题在于所有的对象类都必须实现同样的一些底层函数。
对于第二个问题,可以通过接口继承的方式来避免基类的方法太多。在mangos的实现中就采用了类似的方法,从Object虚基类派生的Unit和WorldObject仍然还是不可实例化的类,这两种对象定义了不同的属性和方法,分来实现不同类型的对象。在游戏内部可以根据对象的实际类型来Object指针向下转型为Unit或WorldObject,以调用需要的接口。方法虽然不够OO,也还能解决问题。但是第一个问题是始终无法避免的。
所以我们便有了通用实体这么一个概念,其主要方法是将原来基类的接口进行分类,分到一个个不同的子类中,然后以对象组合的方式来生成我们所需要的实际游戏对象类型。这个组合的过程可以通过脚本定义的方式,这样便可以在运行时生成为同的对象类型,也就解决了上面提到的第一个问题。
通用实体的实现方法在目前的游戏引擎及开源代码中也可以看到影子。一个是BigWorld,从提供的资料来看,其引擎只提供了一个entity游戏对象,然后由游戏内容实现者通过xml和python脚本来自由定义不同类型的entity类型,每种类型可有不同的property和不同的方法。这样原来由基类定义的接口完全转移到脚本定义,具有非常强的灵活性。
另外还有一个是CEL中的entity实现。按照CEL的描述,entity可以是游戏中的任意对象,包括玩家可交互的对象,如钥匙、武器等,也可以包括不能直接交互的对象,如游戏世界,甚至任务链中的一部分等。entity本身并没有任何特性,具体的功能实现需要靠附加property来完成。简单来说,property才定义了entity可以做什么,至于该怎么做,那又是依靠behavior来定义。所以,最终在CEL中一个游戏对象其实是由entity组合了多个property及多个behavior而生成的。
但是CEL中的property与BigWorld中的property意义不大一样,在CEL中可定义的property其实是引擎内部要先提供的,比如其示例中所举的pcobject.mesh、pcmove.linear、pctools.inventory等,而BigWorld中的property是完全的自由定制。从这个角度来讲,其实可以把CEL中的property看作是游戏的逻辑系统,也就是我们上面所描述的,接口分类后所定义的子类。
由引擎内部提供可选择的property与BigWorld所采用的完全自由定制property其实本质上是相同的。在用BigWorld实现的游戏中,也不可能为每种游戏对象类型都完全从头定义property,基于代码利用的原则,也会先定义一些小类,然后在entity类型定义时以自定义property的方式来包含这些小类。当然,我没有使用过BigWorld,上面的描述也只是基于游戏实现的大原则所做出来的。
描述的依然有些抽象,我们可以用wow及mangos代码来说明一下。mangos中为object定义了一个属性集合,根据对象类型的不同,这个属性集的大小及保存数据也会有差异,另外游戏对象内部含有处理不同游戏逻辑的系统,如RestSystem、FloodFilterSystem、VariousSystem等等,在player.h中以接口组的方式进行定义。
如果要将这种结构改为我们描述的通用entity系统,可以让object只提供property注册和删除的接口,这里的property定义与CEL中的相同,放在mangos中也就是上面说的RestSystem、FloodFilterSystem、VariousSystem这些。然后也通过xml文件的方式定义我们所需要的游戏对象类型,如player,creature,item等,不同的对象类型可以选择加载不同的property,加载的原则是需要哪些功能就加载哪些property。
对象的定义按上面的方法完成后,对象的实现也需要做一些修改。以前客户端的消息是直接交由player来处理,AI也是直接调用creature的接口来完成一些功能,现在通用的entity内部已经没有任何可用的方法,所有的实现都转到了property中,所以需要由各个property实现自己来注册感兴趣的事件及消息,entity实现一个消息的转发,转给对此感兴趣的property来处理。其余的实现就没有什么不同了。
当然,我们再做一点扩展,让property不光由引擎来提供,用脚本本身也能定义property,并且可以通过xml来注册这些property,这样便实现了与BigWorld一样的完全自由特性。这其实也就是将很多用C++实现的功能转移到了python中,这种做法可作为参考,但不一定对所有人合适,至少在我看来,这样的实现也基本只能由程序员来做,所以让程序员选择自己最擅长的语言可能会更易于开发和调试。
有关游戏对象实现的描述,前面两篇文章中说的不甚清楚,主要是一直都要引用网上能够找到的资料来进行描述,以避免与公司引起不必要的麻烦。所以语言有些拼凑的感觉,举的例子也很不恰当,今天正好看到了游戏编程精粹五和六上的两篇文章,内容都差不多,<<基于组件的对象管理>>和<<基于组件的游戏对象系统>>,说的也是我上两篇文章想要描述的内容,所以再补一篇,引用其中的部分文字进行明确的说明。
传统的游戏对象管理系统采用继承的方式来实现,例如,所有的子类都从CObject派生。大多数情况下,直接派生的也是抽象类,其中带一些功能而另一些子类则不带这些功能,比如可控制/不可控制,可动画/不可动画等。mangos的实现中基本就是这种情况,从Object直接派生的Unit和WorldObject都是不可直接实例化的类。
传统方法的问题在于无法应对需求的变化,如要求武器也有动画效果时就无法处理了。如果硬要是这样做,那随着需求的啬,很多的方法会被放到基类中,最终的结果是继承树变得越来越头重脚轻,这样的类会丧失它的内聚性,因为它们试图为所有对象完成所有的事。
就是说到最后,基类会有一个很长的接口列表,而很多的游戏对象类型根本不需要实现其中的一些甚至大部分接口,但是按照这种结构却又必须去实现。以至于于实现一个非常庞大的对象,而且想要修改一点功能会导致系统的大调整。
我们希望的系统是可以将现有的功能组合到新的对象中,并且在将新的功能添加到现有的对象中时不需要重构大量的代码和调整继承树的结构。
实现的方法就是从组件来创建一个对象。组件是一个包含所有相关数据成员和方法的类,它完成某个特定的任务。把几个组件组合在一起就可以创建一个新的对象。如把Entity组件、Render组件和Collectable组件组合在一起生成了一个Spoon对象。Entity组件让我们可以把对象放到游戏世界中,Render组件让我们可以为对象指定一个模型进行渲染,而Collectable组件让我们可以拾取这个对象。
关于组件的实现,所有的组件都从一个基础组件接口派生,可称其为IComponent。每个组件也有自己的接口定义,并且这个接口也需要从IComponent派生,类似于这样:IComponent -- ICmpRender -- CCmpRender
这里的每个组件也就是我在上一篇中所说的由引擎提供的属性,或者说在BigWorld中自己实现然后定义的属性,或者使用mangos中的定义,就是一个个的System,虽然mangos并没有将其完全做成组件,但是通过其代码注释可以看到,接口也是按功能组进行了分类,如果要拆分成组件也是比较方便的。
组件之间的通信有两种方法,一是用组件ID查询到组件接口指针,然后调用接口方法;二是使用消息的方式,向对象中所有组件发消息。在初始化的时候,每一个组件类型都会告诉对象管理器应该接收什么样的消息。
查询接口的方法也就是直接的方法调用,只不过接口不是全部在基类中,所以必须先查询到指定的组件然后才能调用其接口。消息的使用前面已经说过多次,其实现方案也有过说明。
最后是关于游戏对象功能的扩展和游戏对象的定义。需要扩展功能也就是需要实现一个新的组件,或者修改现在组件。在大多数情况下,扩展都不会引起结构的很大调整,受影响的最多只是使用到该组件的部分代码。
游戏对象定义可采用完全数据驱动的方式,使用xml或者脚本语言来定义对象类型,以及每个类型需要加载的组件。对象类型注册到对象管理器后,由管理器提供创建指定类型的对象的方法。数据驱动的方式能够让策划自由定义游戏对象类型,并且随时可自由创建新的对象类型。
LuaForWindows 环境配置
from: http://bbs.luaer.cn/read-Lua-tid-233.html
环境:lua for windows (lfW)
主页:http://luaforwindows.luaforge.net/
lua for windows其实是一整套Lua的开发环境,它包括:
Lua Interpreter(Lua解释器)
Lua Reference Manual(Lua参考手册)
Quick Lua Tour (Lua快速入门)
Examples (Lua范例)
Libraries with documentation (一些Lua库和文档)
SciTE (一个很棒的多用途编辑器,已经对Lua做了特殊设置)
其它详细的内容请到luaforge的主页上查看。
之所以推荐这套环境是因为它整合了在windows学习和开发Lua所需要的所有东西
,对于新手来说是非常体贴的,附带的SciTE只要经过简单配置就能够很方便的编
写Lua程序,编译,运行,调试。它还是附带自动提示和代码自动补全功能的哦,
对于用惯VC + VA的开发人员来说,实在是太亲切了。
下面介绍一下整个lfW:
下载lfw,最新的版本是5.1.3.13,直接安装,注意最后一步会询问SciTE编辑器
是否使用“黑色”风格,我比较喜欢黑色底色,所以在这里打钩,之后继续。
安装完成后,学习Lua所需要的一切环境就全部安装完毕,十分简单。
下面可以测试是否安装成功
打开SciTE,新建一个文件,输入一行lua代码:
print("hello,lua")
然后保存为hello.lua,注意保存文件时要加文件名后缀.lua,否则可能不能正确
的运行。
按F5,如果SciTE的输出窗口出现
>lua -e "io.stdout:setvbuf 'no'" "hello.lua"
hello,lua
>Exit code: 0
字样则代表整个lua开发环境安装成功。
如果对SciTE默认的配色方案或者字体不满意,点击【Options】菜单中的【Open
Global Options File】,则可以看到SciTE环境的全局配置文件。里面可以修改
的包括字体,颜色,窗口布局等等,修改相应的值即可。如果找不到自己想要修
改的项目,可以再到【Options】的【Open black.properties】(如果使用的是
白色背景,这里则是white.properties)文件中查找,这里存储的是更加细致的
属性配置。修改这两个文件,基本上就能够满足大部分学习或是开发者的个人喜
好。还有一点,在Global Options File中,查找command.help.*.lua,后面对应
的是在编辑器中按下F1键弹出的chm格式的lua手册文件路径,这里需要修改一下
,把文件路径改正确就可以了(5.1.3版本似乎没这个问题了)。
整个环境还附带一个QuickLuaTour,是一个用Lua写的一个基于控制台的小教程,
很易于理解。
基本上整个环境就是这样,之后就可以开始学习Lua了。
【推荐】强大的代码阅读工具Understand
4 个附件
_http://www.scitools.com/products/
understand/
Understand软件的功能主要定位于
代码的阅读理解。界面貌似是用Qt开发的。
具备如下特性:
1、支持多语言:Ada, C, C++, C#, Java, FORTRAN, Delphi, Jovial, and PL/M ,混合语言的project也支持
2、多平台: Windows/Linux/Solaris/HP-UX/IRIX/MAC OS X
3、
代码语法高亮、
代码折叠、交叉跳转、书签等基本阅读功能。
4、可以对整个project的architecture、metrics进行
分析并输出报表。
5、可以对
代码生成多种图(butterfly graph、call graph、called by graph、control flow graph、UML class graph等),在图上点击节点可以跳转到对应的源
代码位置。
6、提供Perl API便于扩展。作图全部是用Perl插件实现的,直接读取
分析好的数据库作图。
7、内置的目录和文件比较器。
8、支持project的snapshot,并能和自家的TrackBack集成便于监视project的变化。
小技巧(官网的FAQ里有):
1、设置字体和颜色风格
修改默认字体:Tools -> Options -> Editor -> Default style
修改颜色: Tools -> Options -> Editor -> Styles
2、生成UML类图、调用树图
默认安装的插件不支持这两种图,需要从官网下载插件。
_http://www.scitools.com/perl_scripts/uperl/uml_class.upl
_http://www.scitools.com/perl_scripts/uperl/invocation.upl
放到sti/conf/scripts/local目录下。
然后重新运行,执行 project-> project graphical views -> xxxx可以生成这两种图。
3、更改图的字体
直接修改对应的脚本文件(\Program Files\STI\conf\scripts目录下),在do_load( )函数的对应位置加入如下的设置:
$graph->default("fontname","Consolas","node");
$graph->default("fontsize","10","node");
$graph->default("fontname","Consolas","edge");
$graph->default("fontsize","10","edge");
注意:有的脚本中的作图变量名不是 $graph 而是 $g。
另外一款
代码可视化理解
工具:
http://www.sgvsarc.com/prod_crystalrevs_screenshots.htm
通过c++调用lua 脚本,
环境VC++6.0
lua sdk 5.1
在调用前 先认识几个函数。
1. 调用lua_open()将创建一个指向Lua解释器的指针。
2. luaL_openlibs()函数加载Lua库。
3. 使用luaL_dofile()加载脚本并运行脚本。
4. lua_close()来关闭Lua指向解释器的指针。
5. 调用lua_getglobal()将add()函数压入栈顶,add()为lua函数。
6. 第一个参数x,通过调用lua_pushnumber()入栈。
7. 再次调用lua_pushnumber()将第二个参数入栈。
8. 使用lua_call()调用Lua函数。
9. 调用lua_tonumber()从栈顶取得函数的返回值。
10. lua_pop()移除栈顶的值。
代码
add.lua
1function add ( x, y )
2 return x + y
3end
4
main.cpp
#include <stdio.h>
extern "C" {
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
}
/**//* the Lua interpreter */
lua_State * L;
int luaadd ( int x, int y )
{
int sum;
//函数名
lua_getglobal(L, "add");
//第一个参数压栈
lua_pushnumber(L, x);
//第二个参数压栈
lua_pushnumber(L, y);
//调用函数
lua_call(L, 2, 1);
//得到返回值
sum = (int)lua_tonumber(L, -1);
lua_pop(L, 1);
return sum;
}
int main ( int argc, char *argv[] )
{
int sum;
//创建一个指向Lua解释器的指针。
L = lua_open();
//函数加载Lua库
luaL_openlibs(L);
//加载脚本
luaL_dofile(L,"add.lua");
//调用函数
sum = luaadd( 10, 11);
// print the result
printf( "The sum is %d\n", sum );
//关闭 释放资源
lua_close(L);
return 0;
}
注意问题:
1.工程头文件lua.h等,编译器能找到,可以通过工具来设置头文件路径。
2. 添加lua5.1.lib到Object/library modules列表中。
测试结果
The sum is 21
关于lua的认识
http://www.cppblog.com/expter/archive/2008/12/24/70224.html
Lua脚本语法说明(增加lua5.1部份特性)
Lua 的语法比较简单,学习起来也比较省力,但功能却并不弱。
所以,我只简单的归纳一下Lua的一些语法规则,使用起来方便好查就可以了。估计看完了,就懂得怎么写Lua程序了。
在Lua中,一切都是变量,除了关键字。
I. 首先是注释
写一个程序,总是少不了注释的。
在Lua中,你可以使用单行注释和多行注释。
单行注释中,连续两个减号"--"表示注释的开始,一直延续到行末为止。相当于C++语言中的"//"。
多行注释中,由"--[["表示注释开始,并且一直延续到"]]"为止。这种注释相当于C语言中的"/*...*/"。在注释当中,"[["和"]]"是可以嵌套的(
在lua5.1中,中括号中间是可以加若干个"="号的,如 [==[ ... ]==]),见下面的字符串表示说明。
II. Lua编程
经典的"Hello world"的程序总是被用来开始介绍一种语言。在Lua中,写一个这样的程序很简单:
print("Hello world")
在Lua中,语句之间可以用分号";"隔开,也可以用空白隔开。一般来说,如果多个语句写在同一行的话,建议总是用分号隔开。
Lua 有好几种程序控制语句,如:
控制语句 |
格式 |
示例 |
If |
if 条件 then ... elseif 条件 then ... else ... end |
if 1+1=2 then print("true") elseif 1+2~=3 then print("true") else print("false") end
|
While |
while 条件 do ... end |
while 1+1~=2 do print("true") end
|
Repeat |
repeat ... until 条件 |
repeat print("Hello") until 1+1~=2
|
For |
for 变量=初值, 终点值, 步进 do ... end |
for i = 1, 10, 2 do print(i) end
|
For |
for 变量1, 变量2, ... 变量n in 表或枚举函数 do ... end |
for a,b in mylist do print(a, b) end
|
注意一下,for的循环变量总是只作用于for的局部变量;当省略步进值时,for循环会使用1作为步进值。
使用break可以用来中止一个循环。
相对C语言来说,Lua有几个地方是明显不同的,所以面要特别注意一下:
.
语句块 语句块在C中是用"{"和"}"括起来的,在Lua中,它是用do 和 end 括起来的。比如:
do print("Hello") end
可以在
函数 中和
语句块 中定局部变量。
.
赋值语句 赋值语句在Lua被强化了。它可以同时给多个变量赋值。
例如:
a,b,c,d=1,2,3,4
甚至是:
a,b=b,a -- 多么方便的交换变量功能啊。
在默认情况下,变量总是认为是全局的。假如需要定义局部变量,则在第一次赋值的时候,需要用local说明。比如:
local a,b,c = 1,2,3 -- a,b,c都是局部变量
.
数值运算 和C语言一样,支持 +, -, *, /。但Lua还多了一个"^"。这表示指数乘方运算。比如2^3 结果为8, 2^4结果为16。
连接两个字符串,可以用".."运处符。如:
"This a " .. "string." -- 等于 "this a string"
.
比较运算
比较符号 |
< |
> |
<= |
>= |
== |
~= |
含义 |
小于 |
大于 |
小于或等于 |
大于或等于 |
相等 |
不相等 |
所有这些操作符总是返回true或false。
对于Table,Function和Userdata类型的数据,只有 == 和 ~=可以用。相等表示两个变量引用的是同一个数据。比如:
a={1,2}
b=a
print(a==b, a~=b) --输出 true, false
a={1,2}
b={1,2}
print(a==b, a~=b) --输出 false, true
.逻辑运算
and, or, not
其中,and 和 or 与C语言区别特别大。
在这里,请先记住,在Lua中,只有false和nil才计算为false,其它任何数据都计算为true,0也是true!
and 和 or的运算结果不是true和false,而是和它的两个操作数相关。
a and b:如果a为false,则返回a;否则返回b
a or b:如果 a 为true,则返回a;否则返回b
举几个例子:
print(4 and 5) --输出 5
print(nil and 13) --输出 nil
print(false and 13) --输出 false
print(4 or 5) --输出 4
print(false or 5) --输出 5
在Lua中这是很有用的特性,也是比较令人混洧的特性。
我们可以模拟C语言中的语句:x = a? b : c,在Lua中,可以写成:x = a and b or c。
最有用的语句是: x = x or v,它相当于:if not x then x = v end 。
.运算符优先级,从低到高顺序如下:
or and < > <= >= ~= == .. (字符串连接) + - * / % not #(lua5.1 取长度运算) - (一元运算) ^ |
和C语言一样,括号可以改变优先级。
III. 关键字
关键字是不能做为变量的。Lua的关键字不多,就以下几个:
and |
break |
do |
else |
elseif |
|
end |
false |
for |
function |
if |
|
in |
local |
nil |
not |
or |
|
repeat |
return |
then |
true |
until |
while |
IV. 变量类型
怎么确定一个变量是什么类型的呢?大家可以用type()函数来检查。Lua支持的类型有以下几种:
Nil |
空值,所有没有使用过的变量,都是nil。nil既是值,又是类型。 |
Boolean |
布尔值,只有两个有效值:true和false |
Number |
数值,在Lua里,数值相当于C语言的double |
String |
字符串,如果你愿意的话,字符串是可以包含"\0"字符的(这和C语言总是以"\0"结尾是不一样的) |
Table |
关系表类型,这个类型功能比较强大,请参考后面的内容。 |
Function |
函数类型,不要怀疑,函数也是一种类型,也就是说,所有的函数,它本身就是一个变量。 |
Userdata |
嗯,这个类型专门用来和Lua的宿主打交道的。宿主通常是用C和C++来编写的,在这种情况下,Userdata可以是宿主的任意数据类型,常用的有Struct和指针。 |
Thread |
线程类型,在Lua中没有真正的线程。Lua中可以将一个函数分成几部份运行。如果感兴趣的话,可以去看看Lua的文档。 现在回过头来看看,倒觉得不是线程类型。反而象是用来做遍历的,象是Iterator函数。 如:
function range(n) local i = 0 while(i < n) do coroutine.yield( i ) i = i + 1 end end
可惜的是要继续运行,需要coroutine.resume函数,有点鸡肋。请指教。
|
V. 变量的定义
所有的语言,都要用到变量。在Lua中,不管在什么地方使用变量,都不需要声明,并且所有的这些变量总是全局变量,除非我们在前面加上"local"。这一点要特别注意,因为我们可能想在函数里使用局部变量,却忘了用local来说明。
至于变量名字,它是大小写相关的。也就是说,A和a是两个不同的变量。
定义一个变量的方法就是赋值。"="操作就是用来赋值的
我们一起来定义几种常用类型的变量吧。
A. Nil
正如前面所说的,没有使用过的变量的值,都是Nil。有时候我们也需要将一个变量清除,这时候,我们可以直接给变量赋以nil值。如:
var1=nil -- 请注意 nil 一定要小写
B. Boolean
布尔值通常是用在进行条件判断的时候。布尔值有两种:true 和 false。在Lua中,只有false和nil才被计算为false,而所有任何其它类型的值,都是true。比如0,空串等等,都是true。不要被 C语言的习惯所误导,0在Lua中的的确确是true。你也可以直接给一个变量赋以Boolean类型的值,如:
theBoolean = true
C. Number
在Lua中,是没有整数类型的,也不需要。一般情况下,只要数值不是很大(比如不超过100,000,000,000,000),是不会产生舍入误差的。在WindowsXP能跑的当今主流PC上,实数的运算并不比整数慢。
实数的表示方法,同C语言类似,如:
4 0.4 4.57e-3 0.3e12 5e+20
D. String
字符串,总是一种非常常用的高级类型。在Lua中,我们可以非常方便的定义很长很长的字符串。
字符串在Lua中有几种方法来表示,最通用的方法,是用双引号或单引号来括起一个字符串的,如:
"That's go!"
或
'Hello world!'
和C语言相同的,它支持一些转义字符,列表如下:
\a bell
\b back space
\f form feed
\n newline
\r carriage return
\t horizontal tab
\v vertical tab
\\ backslash
\" double quote
\" single quote
\[ left square bracket
\] right square bracket
由于这种字符串只能写在一行中,因此,不可避免的要用到转义字符。加入了转义字符的串,看起来实在是不敢恭维,比如:
"one line\nnext line\n\"in quotes\", "in quotes""
一大堆的"\"符号让人看起来很倒胃口。如果你与我有同感,那么,我们在Lua中,可以用另一种表示方法:用"[["和"]]"将多行的字符串括起来。(
lua5.1: 中括号中间可以加入若干个"="号,如 [==[ ... ]==],详见下面示例)
示例:下面的语句所表示的是完全相同的字符串:
a = 'alo\n123"'
a = "alo\n123\""
a = '\97lo\10\04923"'
a = [[alo
123"]]
a = [==[
alo
123"]==]
值得注意的是,在这种字符串中,如果含有单独使用的"[["或"]]"就仍然得用"\["或"\]"来避免歧义。当然,这种情况是极少会发生的。
E. Table
关系表类型,这是一个很强大的类型。我们可以把这个类型看作是一个数组。只是C语言的数组,只能用正整数来作索引;在Lua中,你可以用任意类型来作数组的索引,除了nil。同样,在C语言中,数组的内容只允许一种类型;在Lua中,你也可以用任意类型的值来作数组的内容,除了nil。
Table的定义很简单,它的主要特征是用"{"和"}"来括起一系列数据元素的。比如:
T1 = {} -- 定义一个空表
T1[1]=10 -- 然后我们就可以象C语言一样来使用它了。
T1["John"]={Age=27, Gender="Male"}
这一句相当于:
T1["John"]={} -- 必须先定义成一个表,还记得未定义的变量是nil类型吗
T1["John"]["Age"]=27
T1["John"]["Gender"]="Male"
当表的索引是字符串的时候,我们可以简写成:
T1.John={}
T1.John.Age=27
T1.John.Gender="Male"
或
T1.John{Age=27, Gender="Male"}
这是一个很强的特性。
在定义表的时候,我们可以把所有的数据内容一起写在"{"和"}"之间,这样子是非常方便,而且很好看。比如,前面的T1的定义,我们可以这么写:
T1=
{
10, -- 相当于 [1] = 10
[100] = 40,
John= -- 如果你原意,你还可以写成:["John"] =
{
Age=27, -- 如果你原意,你还可以写成:["Age"] =27
Gender=Male -- 如果你原意,你还可以写成:["Gender"] =Male
},
20 -- 相当于 [2] = 20
}
看起来很漂亮,不是吗?我们在写的时候,需要注意三点:
第一,所有元素之间,总是用逗号","隔开;
第二,所有索引值都需要用"["和"]"括起来;如果是字符串,还可以去掉引号和中括号;
第三,如果不写索引,则索引就会被认为是数字,并按顺序自动从1往后编;
表类型的构造是如此的方便,以致于常常被人用来代替配置文件。是的,不用怀疑,它比ini文件要漂亮,并且强大的多。
F. Function
函数,在Lua中,函数的定义也很简单。典型的定义如下:
function add(a,b) -- add 是函数名字,a和b是参数名字
return a+b -- return 用来返回函数的运行结果
end
请注意,return语言一定要写在end之前。假如我们非要在中间放上一句return,那么就应该要写成:do return end。
还记得前面说过,函数也是变量类型吗?上面的函数定义,其实相当于:
add = function (a,b) return a+b end
当重新给add赋值时,它就不再表示这个函数了。我们甚至可以赋给add任意数据,包括nil (这样,赋值为nil,将会把该变量清除)。Function是不是很象C语言的函数指针呢?
和C语言一样,Lua的函数可以接受可变参数个数,它同样是用"..."来定义的,比如:
function sum (a,b,)
如果想取得...所代表的参数,可以在函数中访问arg局部变量(表类型)得到 (
lua5.1: 取消arg,并直接用"..."来代表可变参数了,本质还是arg)。
如 sum(1,2,3,4)
则,在函数中,a = 1, b = 2, arg = {3, 4} (
lua5.1: a = 1, b = 2, ... = {3, 4})
更可贵的是,它可以同时返回多个结果,比如:
function s()
return 1,2,3,4
end
a,b,c,d = s() -- 此时,a = 1, b = 2, c = 3, d = 4
前面说过,表类型可以拥有任意类型的值,包括函数!因此,有一个很强大的特性是,拥有函数的表,哦,我想更恰当的应该说是对象吧。Lua可以使用面向对象编程了。不信?举例如下:
t =
{
Age = 27
add = function(self, n) self.Age = self.Age+n end
}
print(t.Age) -- 27
t.add(t, 10)
print(t.Age) -- 37
不过,t.add(t,10) 这一句实在是有点土对吧?没关系,在Lua中,我们可以简写成:
t:add(10) -- 相当于 t.add(t,10)
G. Userdata 和 Thread
这两个类型的话题,超出了本文的内容,就不打算细说了。
VI. 结束语
就这么结束了吗?当然不是,接下来,我们需要用Lua解释器,来帮助理解和实践了。相信这样会更快的对Lua上手了。
就象C语言一样,Lua提供了相当多的标准函数来增强语言的功能。使用这些标准函数,可以很方便的操作各种数据类型,并处理输入输出。有关这方面的信息,我们可以参考《Programming in Lua 》一书,也可以在网络上直接观看电子版,网址为:
http://www.lua.org/pil/index.html 备注:本文的部份内容摘、译自lua随机文档。
相关链接:
1. Lua 官方网站:
http://www.lua.org 2. Lua Wiki网站,你可以在这里找到很多相关的资料,如文档、教程、扩展,以及C/C++的包装等:
http://lua-users.org/wiki/3. Lua 打包下载(包括各种平台和编译器的工程文件如vs2003,vs2005):http://luabinaries.luaforge.net/download.html
这是我编译好的Lua5.02的解释器:
http://files.cnblogs.com/ly4cn/lua.zip
转载: http://www.d2-life.com/LBS/blogview.asp?logID=41
使用Lua作脚本,主要是因为它小巧玲珑(体积小,运行快),而且它的语法又比较简单明了。不过,使用LuaAPI将Lua引擎集成到程序中,确实有一些不方便——用落木随风网友的话来说,就是"就象用汇编"。当然,现在你不用再这么辛苦了,因为你可以使用LuaWrapper For C++。使用这个工具,在C++中集成Lua脚本就是轻而易举的事。你原有的C++函数和类,几乎不需要任何改变,就可以与Lua脚本共享。
作者: 沐枫 (第二人生成员)
版权所有转载请注明原出处
主页:第二人生 http://www.d2-life.com
http://www.d2-life.com/LBS/blogview.asp?logID=41
为什么要用Lua作脚本?
使用Lua作脚本,主要是因为它小巧玲珑(体积小,运行快),而且它的语法又比较简单明了。不过,使用LuaAPI将Lua引擎集成到程序中,确实有一些不方便——用落木随风网友的话来说,就是"就象用汇编"。当然,现在你不用再这么辛苦了,因为你可以使用LuaWrapper For C++。使用这个工具,在C++中集成Lua脚本就是轻而易举的事。你原有的C++函数和类,几乎不需要任何改变,就可以与Lua脚本共享。
我们接下来,用实例来说明,如何用LuaWrapper来集成Lua脚本到你的程序中去。
1. 创建Lua引擎
LuaWrap lua; 或者 LuaWrap* lua = new LuaWrap;
创建一个LuaWrap对象,就是创建一个Lua脚本引擎。并且根据Lua的特性,你可以创建任意多个Lua引擎,甚至可以分布在不同的线程当中。
2. 装载并执行脚本程序
你可以从缓冲区中装载Lua脚本:
lua.LoadString(
"print('Hello World')"
);
当然,你也可以从文件中装入,并执行Lua脚本:
Lua.LoadFile("./test.lua");
Lua的脚本,可以是源代码,也可以经过编译后的中间代码。也许你对编译后的中间代码更感兴趣——如果你不希望让源代码赤裸裸的袒露在大家的眼前。
3. 获取和设置Lua变量
能够获取和设置脚本变量的内容,是一个最基本的功能。你可以使用GetGlobal和SetGlobal函数来做到这一点:
(1) 获取变量:
int a = lua.GetGlobal<int>("a");
LuaTable table = lua.GetGlobal<LuaTable>("t");
这里,<> 里头的类型,就是想要的变量的类型。
(2) 设置变量:
lua.SetGlobal("a", a);
lua.SetGlobal("t", table);
4. 调用Lua函数
使用Call函数,就可以很简单的从你的程序中调用Lua函数:
lua.Call<void>("print", "Hello World");
int sum = lua.Call<int>("add", 2, 3);
这里,<> 里头的类型是返回值的类型。
5. 如何让Lua也能调用C++的函数
精采的地方来了。假如有下面这样的一个函数:
int add(int a, int b)
{
return a + b;
}
如果想让它能够让Lua使用,只需将它注册到Lua引擎当中就可以了:
lua.RegisterFunc("add", int(int,int), add);
这样,Lua中就可以用直接使用了:
(Lua脚本)sum = add(1, 3)
(*) RegisterFunc的功能,就是让你把C++的函数注册到Lua中,供Lua脚本使用。
第一个参数,是想要在Lua中用的函数名。
第二个参数,是C++中函数的原型; C++允许函数重载的,你可以使用函数原型,来选择需要注册到Lua引擎中的那个函数。
第三个参数,就是C++中函数的指针了。
6. 如何能让C++的类在Lua中使用
我们先看看下面这个C++类:
class MyArray
{
std::vector<double> array;
public:
void setvalue(int index, double value);
double getvalue(int index);
int size();
const char* ToString();
};
你准备要让Lua能够自由访问并操作这个类。很简单,你只需增加几个宏定义就可以了:
class MyArray
{
std::vector<double> array;
public:
void setvalue(int index, double value);
double getvalue(int index);
int size();
const char* ToString();
// 将一个 class 作为一个 Lua 对象是很容易的,只需要增加以下宏定义。
DEFINE_TYPENAME("My.array");
BEGIN_REGLUALIB("array")
LUALIB_ITEM_CREATE("new", MyArray ) // 创建MyArray
LUALIB_ITEM_DESTROY("del", MyArray ) // 消除MyArray。
END_REGLUALIB()
BEGIN_REGLUALIB_MEMBER()
LUALIB_ITEM_FUNC("size", int (MyArray*), &MyArray::size)
LUALIB_ITEM_FUNC("__getindex", double(MyArray*, int), &MyArray::getvalue)
LUALIB_ITEM_FUNC("__newindex", void (MyArray*, int, double), &MyArray::setvalue)
LUALIB_ITEM_FUNC("__tostring", const char* (MyArray*), &MyArray::ToString)
LUALIB_ITEM_DESTROY("__gc", MyArray ) // 垃圾收集时消除对象用。
END_REGLUALIB_MEMBER()
};
只要有了这些宏定义,这个类就是可以在Lua中使用的类了,我们就可以在Lua中注册这个类了:
lua.Register<MyArray>()
这样注册以后,我们在Lua中就可以使用这个类了:
a = array.new() -- 创建对象,相当于 a = new Myarray
a[1] = 10 -- 调用__newindex,也就是C++中的 a->setvalue(1, 10)
a[2] = 20 -- 调用__newindex,也就是C++中的 a->setvalue(2, 20)
print(
a, -- 调用 __tostring,也就是C++中的 a->ToString()
a:size(), -- 相当于C++中的 a->size()
a[1], -- 调用__getindex,也就是C++中的a->getvalue(1)
a[2]) --调用__getindex,也就是C++中的a->getvalue(2)
array.del(a) -- 清除对象,相当于 delete a
a = nil -- 清空 a,很象C++中的 a = NULL
当然,你也可以不用del这个对象,而是等待Lua帮你自动进行垃圾回收。在Lua进行垃圾回收时,它会自动调用这个对象的 __gc ,相当于 delete。
那么,在C++中要创建MyArray对象,并且传递给Lua全局变量怎么办?就象前面讲过的一样,使用SetGlobal:
MyArray* a = new MyArray;
lua.SetGlobal("a", a);
要获取该对象,同样的,应该使用GetGlobal:
MyArray* a = lua.GetGlobal<MyArray>("a");
对于传递给Lua的对象,就让Lua来管理该对象的生存周期好了。如果你非要删除它的话,你可以使用DelGlobalObject:
lua.DelGlobalObject<MyArray>("a");
不过这么做的话,你应当明白你在做什么,因为在Lua的脚本中,可能已经在多处引用了这个对象了。删除了其中一个,将导致其它引用对象失效,从而可能引致系统崩溃。
(1) DEFINE_TYPENAME("My.array");
定义类型的名称。在Lua中,这个类型名称是唯一用来识别C++类型的,你必须为不同的对象给予不同的名称。
(2) BEGIN_REGLUALIB("array") ... END_REGLUALIB()
你可以为一个对象定义一个程序库,"array"就是程序库的名字。在程序库中定义的函数是全局函数,在Lua中,使用该函数,需要在函数前加上库的名字,如:array.new()。通常,程序库会包含创建对象的方法。如:
LUALIB_ITEM_CREATE("new", MyArray ) // 创建MyArray
这样子,你才能在Lua中创建MyArray:
a = array.new()
你也可以选择增加一个删除对象操作:
LUALIB_ITEM_DESTROY("del", MyArray ) // 删除MyArray
这样,你就可以直接删除一个对象了:
array.del(a)
(3) BEGIN_REGLUALIB_MEMBER() ...END_REGLUALIB_MEMBER()
在此处,你可以定义对象的成员函数,也可以重载对象的操作符——是的,就象C++的operator重载。例如:
LUALIB_ITEM_FUNC("__newindex", void (MyArray*, int, double), &MyArray::setvalue)
就是重载 operator[] 操作符。Lua中可重载的操作符还有许多,如:
__getindex:操作符[],支持读取访问,如 v = a[10]
__newindex:操作符[],支持赋值访问,如 a[10] = 1.22
__tostring:将变量转换成字串__add:等同于operator +
__add:操作符 +
__sub:操作符 -
__mul:操作符 ×
__div:操作符 ÷
__pow:操作符 ^ (乘方)
__unm:一元操作符 -
__concat:操作符 .. (字符串连接)
__eq:操作符 == (a ~= b等价于 not a == b)
__lt:操作符 < (a > b 等价于 b < a)
__le:操作符 <= (a >= b 等价于 b <= a,要注意的是,如果没有定义"__le",则Lua将会尝试将a<=b 转换成 not (b < a) )
__gc:在垃圾回收时调用此函数,相当于C++的析构函数。强烈建议定义此操作符,以免造成内存泄漏等情况。比如:
LUALIB_ITEM_DESTROY("__gc", MyArray ) // 垃圾收集时消除对象用。
(注) 这里要说明一下,在lua中,访问索引操作符是__index,不是__getindex,在luaWrapper库中,为了方便使用,将其映射为__getindex,同时,对__index的定义将会被忽略。
就这么简单。假如你已经有现成的类,而你没有修改该类的权力,如何将其加入到Lua中呢?答案就是,继承它,将把派生类加入到Lua中。
结束语
LuaWrapper 需要用到boost库的支持:boost/type_traits.hpp, boost/function.hpp, boost/bind.hpp,它使用了C++的模板部份特化,因此,C++编译器如果不支持此特性,将无法编译。目前支持此特性的编译器已经有很多。在VisualStudo产品系列中,只有VC7.1能支持此特性,因此,您如果正在使用VisualStudio,请确认你用的是VisualStudio2003。
如果你觉得 LuaWrapper For C++ 能够帮助你,我会感觉很荣幸。我很愿意将这个程序库分享给大家。顺便一提的是,如果你在使用过程中发现BUG,或是有好的建议,希望您能与我联系。你在使用过程中,请不要删除文件中的署名信息;如果你修改了程序库,请您在修改的文件中加入您的修改说明。当然,我会非常欢迎您能将修改后的程序回馈给我。我会继续优化并完善它。