H5C3J 学习笔记 part9 Javascript inherits spriteSheet
这几天一直在一个小东西上改来改去,今天终于成了一点形,得马上记录一下,不然就堆积得太多了。
首先是编写好一个Bird类,或者说,Vehicle,又或者说,Atomic类,总之是一个具备了2D向量行为的类型,下面我很自然的顺着原来的思路,想要编写一个该类的子类,然后第一次发现JavaScript是木有继承概念的……啊,我孤陋寡闻了。
我翻看了一些所谓的完美继承方案,大部分都看得很累,这些人写文章并非为了解决你的继承问题,而是炫耀他从过程中获得的点点优越感(我也是,但是没那么严重)。
然后我找到了这里,我认为这个老外用很少的代码,实现了基本完整的继承机制,原文在此
代码如下:
Function.prototype.method = function (name, func) {
this.prototype[name] = func;
return this;
};
Function.method('inherits', function (parent) {
this.prototype = new parent();
var d = {},
p = this.prototype;
this.prototype.constructor = parent;
this.method('uber', function uber(name) {
if (!(name in d)) {
d[name] = 0;
}
var f, r, t = d[name], v = parent.prototype;
if (t) {
while (t) {
v = v.constructor.prototype;
t -= 1;
}
f = v[name];
} else {
f = p[name];
if (f == this[name]) {
f = v[name];
}
}
d[name] += 1;
r = f.apply(this, Array.prototype.slice.apply(arguments, [1]));
d[name] -= 1;
return r;
});
return this;
});
Function.method('swiss', function (parent) {
for (var i = 1; i < arguments.length; i += 1) {
var name = arguments[i];
this.prototype[name] = parent.prototype[name];
}
return this;
});
`</pre>
要定义类型时,先声明一个空白的构造函数,然后声明一个init函数,在里面定义要使用的类变量。
<pre>` var Bird = function ()
{
}
Bird.method("init",function(px,py,vx,vy){
this.edgeBehavior = Bird.BOUNCE;
this.mass = 1.0;
this.maxSpeed = 30;
});
`</pre>
想定义子类时,显式的调用<span>inherits函数,即可实现继承,例如:</span>
<pre>` var Tank = function()
{
}
Tank.inherits(Bird);
Tank.method("init",function(px,py,vx,vy){
//Some code for Tank Class,override parent function init
});
`</pre>
第二个值得注意的地方是,this关键字的引用歧义陷阱,如下代码开始的时候让我完全没有注意到错误的来源:
<pre>` this.reachGoal = false;
setTimeout(function(){this.reachGoal = true},1000);
`</pre>
JavaScript没有类概念,它的所谓类,就是一个函数而已,所以当我使用匿名函数做定时器调用时,这里的this,指代的已经不是我想操作的this,而是指代那个匿名函数了。解决的办法是(我暂且这样解决的)起个别名变量。
<pre>` this.reachGoal = false;
var main = this;
setTimeout(function(){main.reachGoal = true},1000);
下面遇到的就是,如何展现一个爆炸动画,当然可以用粒子系统达成,但个人认为效率低,效果没有那么好,所以还是利用一个老技术,SpriteSheet来达成,可以很容易的找到一些资源,比如下面这种。
[![](https://images.tigerwang.us/hecool108/explosion2.png)](https://images.tigerwang.us/hecool108/explosion2.png)
Collie引擎中对SpriteSheet有很好的支持,当我把这玩意当个新鲜事儿和小白分享时,小白说他早就知道这个东西了,其解决的最大问题,就是可以减少发起http请求的次数,我当时满脑子惊异,在我看来,这东西是为了解决:“如何能象Flash一样利用单个文件描述一个连续动画”的问题,然后我猛然发现,我以Flash为中心的思路是多么狭隘,我考虑这问题时,完全忽略了这项技术可能先于Flash诞生的事实,而且Flash中很可能借鉴了这样的手段,只是它封闭起来,我没看见而已。
我从日本人对美国的态度上,学习到的是,知道人家比你强,就马上认输,俯首学习。
关于转向行为的部分代码,在看的书里目前都有,所以不再赘述。
下面是这几天的一个小小的结果展示。
[![](https://images.tigerwang.us/hecool108/%2525E5%2525B1%25258F%2525E5%2525B9%252595%2525E5%2525BF%2525AB%2525E7%252585%2525A7%2525202013-10-14%252520%2525E4%2525B8%25258B%2525E5%25258D%25258811.12.14.png)](https://images.tigerwang.us/hecool108/%2525E5%2525B1%25258F%2525E5%2525B9%252595%2525E5%2525BF%2525AB%2525E7%252585%2525A7%2525202013-10-14%252520%2525E4%2525B8%25258B%2525E5%25258D%25258811.12.14.png)
[demo](http://hector.ziki.me/tank/tank/Tank.html)
很郁闷的是,发现它在Safari下又不能正常工作了……,检查一下发现,用于实现帧动画的requestAnimationFrame函数在Mac版本的Safari下不能正常运行,在iOS6的Safari下也不能正常运行,在iOS7的Safari上可以运行,所以,我只好再改成利用Timer触发的版本了。
这个水真不是一般的深啊,我发现原来用AS实现的部分,例如层次关系坐标换算,用AS实现时很容易,做js开发果然要痛苦一些。