该教程适用于中等水平,熟悉编程的基本常识,并想进一步扩展自己的知识与能力的开发者.
在开发地图的过程中遇到某些情景,你可能不太想要受限于类型code
,以实现提供自定义的事件监听器,延迟代码的执行,在选取单位时使用一个匹配函数,或者只是简单的,把函数当做对象一样存储在变量中之类的功能.
在Jass中,把函数(类型code
)存储在变量中也是可能的,但终究没法大规模的对他们进行操作,究其原因还是因为jass并不允许code array
,并且你也没法将code
放入哈希表中。
在Jass中的确是存在着一些解决方案去调用和储存函数,这通常需要开发者的一些小技巧,这些技巧通常依赖于boolexpr
和trigger
,或者不安全的ExecuteFunc(string name)
来制作伪函数变量。
而这些缺陷则都被闭包解决了.
但在我们接触闭包之前,为了防止出现概念的混淆,我们先来区分一些东西.
lambda表达式能够允许你定义匿名函数,闭包则与它结合使用.
但什么是匿名函数呢?很简单,没有名字去引用它的函数就是匿名函数.看一下以下例子.
function foo()
print("Hello")
init
CreateTrigger().addAction(function foo)
我们正在给addAction
传递一个函数名作为引用,自然这个函数(作为参数的那一个)是有名字的.或者说,这个函数是非匿名的.
那现在让我们用lambda表达式来实现一个匿名函数的版本.
init
CreateTrigger().addAction(() -> print("Hello"))
新的部分在这儿, () -> print("Hello")
正如你看到的,这就是跟在函数名后面的部分
/*function foo*/()
print("Hello")
通过添加箭头->
来表明方向和函数体的开始. 这在有参数的情况下也适用.
function foo(int i1, int i2)
print("Hello " + i1.toString + i2.toString())
// 可以写成
(i1, i2) -> print("Hello " + i1.toString + i2.toString())
但是这样的情况不能作用于类似addAction
的函数,毕竟在jass中你无法去唤出(invoke)类型为code
的且带参数的函数.你将不得不用全局函数GetXXX
来获取事件相关的或者传递过来的数据.
因此,我们需要闭包.
闭包(closure)使用类来突破这些限制,并让编译器来自动的为你处理这些糟心的的工作.
闭包是通过隐式创建的,这意味着你不会去调用new
来实例化它们.要用闭包,你得先定义一个构建你的闭包对象的类.
让我们来举个例子,来实现一个有两个整型作为参数的回调函数. 通过实现一个包含单一抽象函数的接口,或者抽象类,我们就能做到这一点了.
我们如期的使用这个类型,目前为止没有什么特别的地方. 让我们来快速测试一下.
interface Calculator
abstract function doCalc(int i1, int i2)
class IntGenerator
let calculators = new LinkedList<Calculator>
function doCalculations()
let i1 = GetRandomInt(-100, 100)
let i2 = GetRandomInt(-100, 100)
print("i1: " + i1.toString() + " i2: " + i2.toString())
for calculator in calculators
print(calculator.doCalc(i1, i2).toString())
function addCalculator(Calculator calculator)
calculators.add(calculator)
@Test
function testCalculations()
let gen = new IntGenerator()
// gen.addCalculator()
gen.doCalculations()
在这我们用了一个 IntGenerator
来接收Calculator
对象,以比较两个随机数在不同的计算方法上的结果.
如果我们不用闭包,我们会需要声明一个类来实现Calculator
接口,重写doCalc
函数,然后传递一个通过new
而实例化的对象给 addCalculator
像这样:
//不用闭包——重写计算器接口的计算方法
class MyCalculator implements Calculator
override function doCalc(int i1, int i2)
return i1 + (i1 * i2)
...
gen.addCalculator(new MyCalculator())
这其实就是Wurst会悄悄替你做的事情.来看下面,函数名后面的部分转换为lambda表达式:
//闭包——lambda表达式写法
let gen = new IntGenerator()
gen.addCalculator() (int i1, int i2) ->
return i1 + (i1 * i2)
gen.doCalculations()
你会注意到,这儿再也没有类MyCalculator
了,但是Wurst会为你生成一个类似的类,悄咪咪的在后台干的.由于你没有名字去访问它,就如同之前匿名函数一样,这也是匿名的.某种程度上你可以把闭包当成匿名的类,每一个对接口的新的实现就会新建一个这样的闭包.
顺带一提,由于interface Calculator
中函数的参数类型已经给定了,我们可以在lambda中忽略这些参数类型.
//闭包——lambda表达式写法【忽略参数类型】
let gen = new IntGenerator()
gen.addCalculator() (i1, i2) ->
return i1 + (i1 * i2)
gen.addCalculator() (i1, i2) ->
return i1 + i2
gen.doCalculations()
如果我们添加了两个计算器(Calculator),Wurst会为我们生成两个闭包.三个的话就会生成三个. 这便已经超越了code
的可能性.不过额外的,我们还可以通过使用类来保留数据.
你通常会想着把外部的数据传递给匿名函数,一个比较常规的例子是延迟执行代码。 Often you want to pass data from the outside into the anonymous function. A common example for this is the delayed execution of code.
比方说你想创建一个类似地狱火的技能,先表演一段彗星从天而降的动画.当然,伤害以及其他的潜在事件必须得发生在一段时间之后————在彗星的特效撞击地标之后.在vjass中你或许会使用类似于TimerUtils以及一个数据结构来做到这一点.
struct DelayData
unit caster
int abilLvl
real x
real y
function doEffect() takes nothing returns nothing
local timer tim = GetExpiredTimer()
local DelayData data = (DelayData) tim.getData()
// Do effect..
endfunction
function onSpell() takes nothing returns nothing
local timer tim = getTimer()
local DelayData data = DelayData.create()
// Assign data..
tim.setData((int)data)
TimerStart(tim, 0.3, false, function doEffect)
endfunction
如果你有好几年的jass编程经验,这对于你来说也许已经司空见惯了,不过这段代码会产生一些问题. 首先是牵涉了开发者大量的手动工作,因此更容易犯错.其流程也是反直觉的自底向上.而Wurst可以替你做这些事情.
使用CLosureTimers
,例子如下.
function onSpell()
let caster = GetSpellAbilityUnit()
let lvl = caster.getAbilityLevel(SPELL_ID)
let target = getSpellTargetPos()
doAfter(0.3) ->
caster.damageTarget(.., lvl * 100)
flashEffect("xxx", target)
在这个场景的背后,wurst做了和你自己在vjass中做得一模一样的事情.创建一个类,实例化它,将它绑定在计时器上,只是这儿用了Wurst版本的TimerUtils
.在wurst的包中可以看到这是如何实现的. .
正如你在IntGenerator
中看到的,闭包类型可以轻松的保存在变量,类和哈希表中.常见的使用情景是在周期性的计时器或者临时事件的回调中,你想要在它的回调函数中去销毁它.一个简单的例子是在10秒内单位死亡则召唤生物中:
function spellEffect()
let listener = EventListener.add(EVENT_UNIT_DEATH) ->
print("Unit died, summoun spawn for caster")
createUnit(...)
doAfter(10.) ->
// Buff over
destroy listener
如你所见我们在一个变量中保存了EventListener.add
返回的监听器,以在10秒后销毁它. 原文:As you can see we save the listener returned by EventListener.add
in a variable to destroy it after 10 seconds.
记住闭包不会被自动的回收,除非底层系统替你做了.标准库会销毁计划中为临时性的闭包.
如在导入中提及的那一,闭包也是一个触发器的替代品,因为触发器在jass中也被用来提供自定义事件. 代替掉使用addAction
并提供回调函数的函数名作为参数的方法,我们定义一个闭包接口/类,并按需保存.
想一下在一张地图中,我们通过Level
类表达大量的关卡.而我们想要注册一些在通关后的自定义的事件.
你可以复制粘贴这个包到你的地图中去,看一看打印输出.
package Level
//类似js的事件监听器,监听完成事件
interface LevelFinishedListener
function onFinish()
//创建关卡类,每1关(1个实例)都能注册【事件—关卡完成】时候的【动作】
class Level
private LevelFinishedListener listener
//添加实例完成监听器【添加动作】
function addFinishListener(LevelFinishedListener listener)
this.listener = listener
//完成关卡,运行【完成动作】
function finishLevel()
print("Level finished")
listener.onFinish()
init
let lvl1 = new Level()
..addFinishListener() ->
print("You beat level 1 - prepare to die")
let lvl1 = new Level()
..addFinishListener() ->
print("You beat level 2 - how unexpected")
// To test the listener
lvl1.finishLevel()
我们希望该教程能帮助你更进一步理解这一高级话题.来看看manual section 获取更多信息吧.