单机与联机的差异和转换

饥荒分为单机版和联机版,一般来说,内容相同的MOD,联机版相对单机版来说,总是要多一些对网络数据的处理,总归是要复杂一些。越复杂的MOD,从单机移植到联机就越困难。

本文主要讲述单机和联机的差异,以及从单机转入联机时,应该注意的东西。

主要资料来自Klei官方论坛的长篇教程:[Guide] Getting started with modding DST (and some general tips for DS as well)
原作者是rezecib,当然,也结合了自己做MOD积累的经验。

相同的部分

相通之处和细节差异

  • 调用美术资源的方式,都是使用assets表和Asset函数。不过在联机版里,如果你要使用自定义的美术资源,还需要在在modinfo里添加语句all_clients_require_mod = true,这条语句会告诉系统要求所有加入房间的人都下载这个MOD。
  • 大部分在单机里可用的MOD API 如AddPrefabPostInit,AddComponentsPostInit等,用法都是不变的。有一点需要特别注意的是,AddSimPostInit,这个函数在单机版里常常被用于在游戏开始之后对玩家进行修改(比如说在游戏开始之后,在玩家附近生成一个专属宠物),这个函数在联机版里不能这样用了,因为当它被执行的时候,还没有玩家生成,所以你应该根据情况,选择新的方法了。如果你想修改UI,用AddClassPostConstruct对playerhud或contraols这两个widget进行修改。
  • 不要再使用AddSimPostInit来修改UI了,在单机时这样做可能仅仅是不太合适,但在联机里很可能会造成游戏崩溃。如果要修改UI,请使用上面提到的AddClassPostConstruct。
  • 添加新动作,单机和联机有些许的不同,具体请参考我这个博客里的另一篇关于如何添加新动作的教程。

让你的MOD在两个游戏中都能运行

一般来说,不推荐在两个游戏中运行同一个MOD,这不方便维护,我建议针对两个游戏发布各自的版本。不过如果你非要这么做的话,也是可以的。方法是使用TheSim:GetGameID()来区分两个游戏要执行的不同的代码块。

在联机版中,TheSim:GetGameID()的返回值是"DST",可以用TheSim:GetGameID() == "DST"这个语句来判断当前运行MOD的游戏是单机版还是联机版。

此外,你还需要在modinfo添加mod api的版本号。单机的api版本为6,联机则为10。写法就是如下:api_version = 6;api_version_dst = 10 。

新的东西

变化

GetWorld()和GetPlayer()

在单机版中存在的两个重要而且常用的全局函数——GetWorld()获取世界和GetPlayer()获取玩家,在联机版中,被改为了全局变量TheWorld和ThePlayer,在联机版中,直接调用这两个变量就可以,不再需要运行函数了。需要注意的是联机版是多人游戏,直接用ThePlayer可能会对其他玩家造成误操作。所以你在改单机为联机时,不可直接将GetPlayer()改为ThePlayer就完事了,而是要仔细分析每个GetPlayer()的使用情景来决定怎样替换。

TheNet

--
联机版新增了一个网络类:TheNet,用于处理各种与网络相关的内容,常用的函数有:

TheNet:GetIsServer() -- 判断是否是主机(创建游戏者)
TheNet:GetIsClient() -- 判断是否是客机(加入游戏者)
TheNet:IsDedicated() -- 判断是否是服务器
TheNet:Announce(message) -- 发送服务器公告,典型例子是XX死于XXX
TheNet:Say(message, whisper) -- 在聊天框里显示信息,如果whisper的值为true,则这个消息只会被附近的人看到

另外,有些人喜欢用TheWorld.ismastersim来代替TheNet:GetIsServer() (还有"not TheWorld.ismastersim" 代替TheNet:GetIsClient()),这在大多数时候没有错,不过碰上某些时候,TheWorld还没创建的情况,就会产生报错,因此还是建议使用TheNet:GetIsServer()。
TheNet类当然也有其它的一些函数,不过最常用的还是上面介绍的那几个,如果想要了解更多,可以查阅游戏目录\data\sripts\screen下的playerstatusscreen.lua找到其他的一些函数,不过这些函数对MOD而言并没有什么卵用(TheNet类不是用lua写的,所以找不到函数的定义,只能看到使用它们的例子)。

modinfo

联机版的modinfo,相比单机版,添加了一些新的内容。

api_version = 10 --api版本,联机版目前为10,如果不填10,则会被提示mod已经过期。
api_version_dst = 10 -- 这个是可选的,如果你想要让这个MOD同时在单机和联机上都能运行,则需要添加这句和下面的那句,并把上面那句的赋值改为6,如果只打算在联机版运行,可不写此句和下面一句。
dst_compatible = true --此值为true则表明MOD兼容联机版
all_clients_require_mod = true--让客机知道自己是否需要订阅这个MOD并下载下来,通常,有自定义美术资源的,都需要令此项为true。
client_only_mod = false--这一项和上一项正好相反,如果此项为true,则这是一个客机MOD,可以在游戏mod界面启动。否则,需要在开启房间的界面启动。
server_filter_tags = {"character", "rog", "reign of giants"}--服务器过滤标签表,主要用于玩家搜索房间。一个房间如果开启了这个mod,就会加入这个mod标签表里的标签。

prefab

联机版的prefab,相比单机版,要多一些额外的关于网络处理的部分。

  • AddNetwork
   inst:AddNetwork() 

当你添加上面这条语句时,就会让这个prefab能被所有人看见(如果它本身是可见的)。否则,只能被创建这个物品的电脑看到。比如说,几何种植显示的那些方格,其实也是prefab,但没有添加上面这条语句,所以只能被种植者本人看到,不会被其他玩家看到。

  • 主机判断
   if not TheWorld.ismastersim then
       return inst
   end

这个if判断块在几乎所有的prefab中都十分重要,代码的含义是,如果这段代码运行在客机上,那么就到这里就结束prefab的初始化了。
因为在联机里,客机上存在的组件是极少的。包括物品栏组件inventoryitem,人物三围hunger,sanity,health在内的众多组件都是不存在的,所以如果不上面的那段代码,客机就会运行执行后面的代码,但由于相应的组件不存在,就一定会造成客机游戏崩溃,这也是单机转联机的人普遍会遇到的第一个崩溃问题。相应的,Tag,以及其他会被所有玩家看到的内容,比如说动画表现(AnimState),形态(Transform),人物说话的文字(talker)等,就必须加在上面这段代码之前,否则客机就看不到相应的界面表现了。

另外,对联机版的人物,初始化函数分成了两部分,一部分是common_postinit,另一部分是master_postinit。前者是会在主机和客机上都执行的代码,后者则只在主机上执行。上面的if判断已经内置在了MakePlayerCharacter函数中,所以无需再添加。

  • SetPristine
   inst.entity:SetPristine() 

这条语句的意思是这条语句执行之前的内容都是运行在主机和客机上的,然后再往下的内容就不要再动用网络了。理论上可以省一点网络带宽,然而这东西很难感觉得出来,也很难做测试,权当没啥卵用吧,一般紧邻着上面的if判断写。写在之前还是之后无所谓。

prefab新内容的实例代码释义

下面给出联机版长矛的初始化函数的代码,参照着上面讲解的内容,很容易就能理解你需要为一个联机版的prefab额外添加哪些东西。

local function fn() 
    local inst = CreateEntity()
    inst.entity:AddTransform()  
    inst.entity:AddNetwork()    --添加联网组件,要想让这个prefab被其他人看到,就必须添加这句,相反的,如果不希望被看到,比如说建筑几何学的网格,就不要加这句。
    MakeInventoryPhysics(inst)    
    inst.AnimState:SetBank("spear")    
    inst.AnimState:SetBuild("spear")   
    inst.AnimState:PlayAnimation("idle")
    if not TheWorld.ismastersim then return inst end    --这个if判断区分了客机和主机
    inst.entity:SetPristine()--节约一点网络带宽
    inst:AddComponent("weapon")    
    inst.components.weapon:SetDamage(TUNING.SPEAR_DAMAGE)        
    inst:AddComponent("finiteuses") 
    inst.components.finiteuses:SetMaxUses(TUNING.SPEAR_USES) 
    inst.components.finiteuses:SetUses(TUNING.SPEAR_USES) 
    inst.components.finiteuses:SetOnFinished(inst.Remove)
    inst:AddComponent("inspectable")
    inst:AddComponent("inventoryitem")
    inst:AddComponent("equippable")
    inst.components.equippable:SetOnEquip(onequip)
    inst.components.equippable:SetOnUnequip(onunequip)
    MakeHauntableLaunch(inst)
end

主客机数据交互

如果你想要做复杂的,深入游戏核心的联机版MOD,主客机的数据交互几乎是不可能绕过去的。如果处理得不好,造成主客机数据不同步,轻则无法顺利实现自己想要的功能,重则直接崩掉服务器(有一段时间,精灵公主联机版不稳定,经常崩服务器,就是因为没有处理好这个问题。如果你想要做复杂的MOD,又想保证一定的可靠性,那么就必须要深入理解这一段的知识。

基本机制

联机版的主机,就和单机版一样,有着几乎所有的prefab,component,stategraph和brain。但在客机上,prefab依然存在,但它们只会运行其中一小部分代码(在TheWorld.ismastersim的判断块之前的代码)。这一小部分代码,只会给prefab添加少量与界面交互有关的组件,比如动画、图片或者文字表现,或者是直接关系者游戏操作的,比如:talker,transparentonsanity,playercontroller, 和playeractionpicker。brain和stategraph都只存在于主机上。相应的,为了使得大多数组件也能在客机上流程地运转,游戏在客机上使用了replica和classified(为什么不直接使用组件?因为这会造成数据的歧义性。比如说一个人物,吃了食物之后,会加饱食度。吃东西这个行为是发生在客机上的,如果客机有Hunger组件,那么调用客机的hunger组件,增加饱食度,那么在客机上,这个人物的饱食就变化了,但主机上的饱食并没有变化,两者的数据就不一致了。为了解决这个问题,就需要把数据都放在主机上,客机只负责接收来自主机的更新数据。)

主客机之间的数据交换,则是通过一个双重系统——remote procedure call(RPC)和netvar来操作。客机通过发送RPC码向主机发出要求。主机则改变netvar的值来向客机传递数据(大多数被储存在了classified中)

Replica

Replica是component的副件,与component不同,不管是主机还客机,replica都是必定会存在的,replica的主要用途就是帮助客机玩家流畅地完成原本component要完成的操作,但这些操作通常都只是游戏界面的变化,比如说播放动画,显示文字之类的,较少涉及到与主机的数据交换(数据交换的工作,主要由classified完成)。在主机中调用component的函数,如果在replica中存在同名函数,也会被同时调用。利用这个特性,可以在同名函数的方法体中。对主机,执行更新客机数据的代码,对客机,执行动画之类的操作(如果有必要的话)。(对主客机执行不同的代码,可以用上面提到的TheWorld.ismastersim这个变量来区分,或者用TheNet:IsServer()这个函数。)

如果你想为自己自定义的新组件设置replica的话,只需要两步便可完成。
1、在components文件夹下,新建一个文件,文件名为"组件名_replica.lua",文件里的定义格式同一般的组件,内容则自行决定。
2、在modmain中,添加一行代码AddReplicableComponent("组件名")
这样一来,你便有了一个自定义的replica。
具体的实例,请参看我的联机版samansha,里面有一个sa_car的replica。如果觉得看不懂,可以在评论区回复,我会考虑专门写一个教学用的实例mod。

Netvar&Classified

Classified和netvar的联系非常紧密,要解释Classified就需要解释netvar,这里就一并讲了。

Netvar,正如其名,是网络变量的意思。Netvar是官方设计用来从主机向客机传递数据的一组数据类型,它们的定义和用法,在netvar.lua里有官方的说明。官方也写过一个教程并配上了相应的示例MOD。不过我觉得官方写得并不够清晰,让我当初学习的时候走了不少弯路。这里按照我自己的理解来写一下,希望能帮到各位读者。

  1. 声明网络变量
    对于有C或java之类强类型语言基础的读者来说,都很清楚声明变量时必须要声明数据类型。netvar也一样。netvar有多种数据类型以供不同规模的数据传输需要,比如net_bool-1位(true,false),net_tinybyte-3位(0-7)等等,具体可以在netvar中查到。在实际使用中,为了节约带宽,我们应当尽量选小数据类型来用。声明一个网络变量的格式为:ReferenceName = NetvarType(entity.GUID, "UniqueName", "DirtyEvent" 。ReferenceName是引用名,NetvarType就是网络变量的数据类型,entity.GUID就是这个网络变量要依附的entity的GUID,一般来说,这个网络变量是谁的,就依附谁。比如说你把这个网络变量写在了一个人物的初始化函数里,那么这个entiy就是inst。UniqueName是唯一名,只需要保证你的网络变量不与其它网络变量重名就行,DirtyEvent是当这个网络变量的数据发生改变时,会触发的事件,一般称为这个网络变量的dirty事件。Dirty事件会在主机和客机上都触发,客机可以利用这一个事件来改变HUD的状态,对做自定义的UI非常有用。

一个示例:inst.level = net_smallbyte(inst.GUID,"MyLevel","leveldirty")

另外,网络变量必须要在主机和客机中都有声明,也就是说,如果你想给人物添加一个网络变量,必须要写在common_init函数里。写在其它prefab里,则要在TheWorld.ismastersim的判断语句块之前写。如果要写在组件里,则必须要保证组件也在客机上存在(否则你应该写在replica里)。

  1. 使用网络变量
    网络变量不可直接赋值,也不可直接读取,需要调用函数来完成。有三个函数可以用,分别是set,value和set_local,具体用法如下,
 netvar:set(x)--只能在主机端调用这个函数,会自动同步客机的数据(在一个新的同步周期开始时)。如果这个函数确实改变了netvar的值,会在主机和客机上都触发相应的dirty事件。
 netvar:value()--可以在主机和客机上调用这个函数,读取当前网络变量的值。

  netvar:set_local(x)--可以在主机或客机上调用,改变相应的值但不触发数据同步或dirty事件。但当主机下一次调用set函数时,无论变量的值是否发生了改变,都会同步一次数据。

同样的,我的联机版samansha也包含了网络变量,仍然是那个sa_car组件及其replica。我主要用到的是dirty事件,对于数据传输,因为没有使用需求,就没有写。不过我会考虑以后加上一个示例的MOD。

Netvar就讲解这么多。下面简单来说一下Classified。Classified,实质上是一个prefab。这个prefab,根据使用的需求,打包了一些联系比较紧密的网络变量,并为他们设置一些共用的函数,以方便被调用(主要是被replica调用)。Classified有好几个,和人物联系紧密的是player_classified,其中包含了关联着饥饿、精神、血量等变量的网络变量。这里不详细展开。对于Classified,主要的用途是,当我们想让客机获取诸如饥饿之类的属性,进行一些与主机无关的操作时,去调取player_classified里的关联hunger的网络变量。但实际上这个应用的范围很小,因为要用到数据,又与主机无关的,一般就剩下控制HUD表现了。大多数时候,我们想要的是,在客机上,呼叫主机执行某一段代码。比如说呼叫主机让hunger减少50点。这个就属于客机向主机发起沟通的范畴了,而这就需要使用RPC。

RPC

RPC是Remote Procedure Call的缩写,意思就是远程程序调用。这里不需要了解其内在机理。只需要知道它是客机用来向主机发送执行代码信号的工具就行了。重点在于学习如何使用它。

RPC是不能随意发送整一段的代码给主机的,它只能发送一条简短的RPC消息,然后主机根据这条消息,寻找相应的RPC处理器,如果找得到的话,就会执行处理器预先设置的代码块。所以,我们想要使用RPC,就需要先(在主机)用AddModRPCHandler函数添加一个RPC处理器,然后在需要的时候,执行SendModRPCToServer来发送一条RPC消息给主机。
我们通过一个例子来学习:

local function GrowGiant(player)    
    player.Transform:SetScale(2,2,2)
end
AddModRPCHandler(modname, RPCname, GrowGiant)--添加RPC处理器,这个语句可以写在任何会被主机执行到的地方。三个参数,第一个为命名空间的名字,建议写mod的名字;第二个参数为RPC的名字,必须是唯一的,如果有同名的,就会根据先后顺序,被后面的覆盖。这两个参数都必须是字符串。第三个参数则是要执行的函数,这个函数的第一个参数固定为玩家的引用,这里的玩家,指的是执行下面的Send

local function SendGrowGiantRPC()   
    SendModRPCToServer(MOD_RPC[modname]["RPCname"])--[[向主机发送RPC消息,第一个参数为MOD_RPC[modname]["RPCname"],MOD_RPC是不能修改的,后面的modname,RPCname就和上面的意思是一样的。另外,这个函数可以传入更多的参数,只要写在第一个参数后面就行。这些参数讲会被上面的AddModRPCHandler里的执行函数接收到。在某些时候会非常有用。
end
GLOBAL.TheInput:AddKeyDownHandler(118, SendGrowGiantRPC)--按v键执行上面定义的SendGrowGiantRPC函数。除了这里通过按键来触发外,我们还可以根据情况,灵活调用SendModRPCToServer来达到不同的目的。

同样的,我的联机版samansha的sa_car组件,也展示了改如何使用RPC,也是可以参考的对象。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容

  • 这一章讲解Prefab。Prefab是饥荒世界构成的基础,也是Mod技术的基本内容。 Prefab,中文译名叫预制...
    LongFei_aot阅读 15,212评论 5 37
  • 这篇文章将阐述如何让你的函数能够在自己设置的限定条件下,通过外部输入(主要是鼠标)来触发执行。 对于很多有了一定M...
    LongFei_aot阅读 5,230评论 0 17
  • 这个系列教程很长,涉及到很多编程、游戏方面的概念。与其让你云里雾里地看着看着就放弃,不如先教你如何快速入门,通过模...
    LongFei_aot阅读 12,539评论 9 73
  • 本章的目的是为了帮助你了解Mod知识的全貌,把握学习的节奏。如果你急着学习更多的技术知识,可以先跳过本章,不会影响...
    LongFei_aot阅读 3,968评论 3 27
  • This article is a record of my journey to learn Game Deve...
    蔡子聪阅读 3,768评论 0 9