- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在制作一款带有定时关卡的游戏。一关时长2分钟,右上角显示倒计时器。马里奥风格。
跟踪已用时间的一种方法是让我的 Level
对象将 elapsed
作为成员变量。
Level.init = function(){
this.elapsed = 0; //member variable!
return this;
};
Level.update = function(){
this.countdown();
};
Level.countdown = function(){
this.elapsed += 1;
var remaining = (GAME_LENGTH*60) - (this.elapsed/TICKS);
var minutes = Math.floor(remaining/60);
var seconds = Math.floor(remaining%60);
this.countdown.html(minutes + ":" + ("0"+seconds).slice(-2));
if (this.elapsed/TICKS >= GAME_LENGTH*60) {
this.level_end();
}
};
当游戏引擎需要新的关卡时,它会以这种方式创建一个:
NewLevel = Object.create(Level).init();
但是,我认为使用闭包可能会更好; elapsed
仅由 countdown
函数使用,因此不需要是 Level
的成员变量。 Level
不需要知道 elapsed
存在。
//member variable this.elapsed has been removed
Level.init = function(){
return this;
};
Level.update = function(){
this.countdown();
};
//a closure is used to keep track of elapsed time
Level.countdown = (function(){
var elapsed = 0;
return function() {
elapsed += 1;
var remaining = (GAME_LENGTH*60) - (elapsed/TICKS);
var minutes = Math.floor(remaining/60);
var seconds = Math.floor(remaining%60);
this.countdown.html(minutes + ":" + ("0"+seconds).slice(-2));
if (elapsed/TICKS >= GAME_LENGTH*60) {
elapsed = 0;
this.end_level();
}
};
})();
但现在我有一个不同的问题。创建的所有 Level
副本都共享该 countdown
闭包,因为它位于原型(prototype)链中。当用户提前退出关卡,然后启动新关卡(废弃旧的 Level
对象并实例化新的 Level
对象)时,elapsed
闭包中的变量不会重置。
换句话说,用户在还剩 30 秒时提前退出关卡。然后用户再次启动关卡,但现在倒计时器仍然显示 30 秒,而不是整整 2 分钟。
有没有一种优雅的方法来解决这个问题并仍然使用闭包?或者我必须恢复到以前的解决方案,忘记闭包,并将 elapsed
设为成员变量?
最佳答案
Is there an elegant way to fix this and still use a closure? Or must I revert to my previous solution, forget closures, and make elapsed a member variable?
它必须是特定于实例的;这并不意味着它必须是对象的属性。您可以在 init
方法中创建闭包:
//member variable this.elapsed has been removed
Level.init = function(){
//a closure is used to keep track of elapsed time
this.countdown = createCountdownMethod();
return this;
};
Level.update = function(){
this.countdown();
};
function createCountdownMethod(){
var elapsed = 0;
return function() {
elapsed += 1;
var remaining = (GAME_LENGTH*60) - (elapsed/TICKS);
var minutes = Math.floor(remaining/60);
var seconds = Math.floor(remaining%60);
this.countdown.html(minutes + ":" + ("0"+seconds).slice(-2));
if (elapsed/TICKS >= GAME_LENGTH*60) {
elapsed = 0;
this.end_level();
}
};
}
或者这可能更容易阅读:
//member variable this.elapsed has been removed
Level.init = function(){
var elapsed = 0;
//a closure is used to keep track of elapsed time
Level.countdown = countdown;
return this;
function countdown() {
elapsed += 1;
var remaining = (GAME_LENGTH*60) - (elapsed/TICKS);
var minutes = Math.floor(remaining/60);
var seconds = Math.floor(remaining%60);
this.countdown.html(minutes + ":" + ("0"+seconds).slice(-2));
if (elapsed/TICKS >= GAME_LENGTH*60) {
elapsed = 0;
this.end_level();
}
}
};
Level.update = function(){
this.countdown();
};
关于javascript - 不需要的 JavaScript 效果 : Prototype is sharing closure between instances,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28730901/
我知道我的问题有点含糊,但我不知道如何描述它。我问过很多地方,但似乎没有人理解我为什么要这样做。但请耐心等待,我会解释为什么我想要这样的东西。 我使用 Liquid Templates 允许用户在我的
这个问题在这里已经有了答案: what is the difference between null != object and object!=null [duplicate] (2 个回答) 7年
当我在我的本地主机 Google App Engine 应用程序中将日志记录级别更改为 FINE 时,我开始在我的跟踪堆栈中看到这些: Apr 17, 2013 4:54:20 PM com.goog
Python 有内置函数 type : class type(object) With one argument, return the type of an object. The return v
我正在使用深度学习进行语义分割,我遇到了以下术语:语义分割、实例检测、对象检测 和对象分割. 它们有什么区别? 最佳答案 这些术语的某些用法对用户而言是主观的或依赖于上下文,但据我所知对这些术语的合理
我面临 -[NSConcreteMutableData release] 的问题:消息发送到已释放的实例,我也附上了我的示例代码。 - (IBAction)uploadImage { NSString
我试图显示模型中的单个实例(数据库行),其中多个实例共享多行的相同字段(列)值。为了澄清这一说法,我有以下情况: ID/Title/Slug/Modified 1 Car A 1s ag
我正在尝试使用mockito来模拟服务。然而,我没有找到一种方法来告诉mockito,给定一个类的实例返回给我相同的实例: 类似于: given(service.add(any(Individua
我知道如何从父类(super class)原型(prototype)创建子类原型(prototype)。但是,如果我已经有了父类(super class)对象的实例来创建子类对象怎么办? 在 JS 中
鉴于 Kotlin 1.1。对于某个类的 instance,instance::class.java 和 instance.javaClass 似乎几乎是等价的: val i = 0 println(
这个问题在这里已经有了答案: 8年前关闭。 Possible Duplicate: Find out the instance id from within an ec2 machine 我正在寻找从
为什么我的 Instantiate 函数没有创建 That 的“空白”实例? 我有以下最小类: classdef That < handle properties This = '' end
Session session = HibernateUtil.getSessionFactory().openSession(); Transaction tx = session.beginTra
考虑以下几点: public class A { public String name = "i am a A instance"; } public class B extends A {
我正在使用 Scalr 来扩展网站服务器。 在 Apache 服务器上,我安装了 Sakai,并为 Linux 机器创建了一个启动脚本。 问题是,如何确保MySQL实例在Apache服务器启动之前启动
Android Realm DB 允许使用 Realm.getInstance() 获取多个实例。这些中的最佳实践是什么? :1.创建单个实例(应用程序范围)并在任何地方使用它2. 在需要时获取一个新
我很难理解为什么修改实例 a 中的属性会修改实例 b 中的相同属性。 var A = function (){ }; A.prototype.data = { value : 0 }; var
我将 Weka 用作更长管道的一部分,因此,我无法承受将所有数据写入文件或数据库只是为了创建一个 Instances。目的。我可以即时做的是创建 Instance 的列表对象。 来自 this pag
class C: def func(self, a): print(a) c = C() print(c.__dict__) # {} c.func = c.func # c.func i
Angular Routing 文档提到了组件实例创建、组件实例激活和路由激活。 文档没有解释这些概念的区别,以及每次创建/激活发生的时间。 问题 实例创建和实例激活有什么区别? 实例激活和路由激活有
我是一名优秀的程序员,十分优秀!