function print(msg){
document.write(msg,"
")
}
function distance(x1,y1,x2,y2){
var dx=x2-x1;
var dy=y2-y1;
return Math.sqrt(dx*dx+dy*dy);
}
function factorial(x){
if(x<=1) return 1;
return x*factorial(x-1);
}
print("hello");//不包含return 返回undefined
factorial(5);//5*4*3*2*1=120
distance(0,0,2,2);//2倍根号2
distance(0,0,2);//NaN 因为y2为undefined
//嵌套的函数
function outerFun(){
function innerFun(x){
return x*x;
}
return Math.sqrt(innerFun(a)+innerFun(b));
}
function f(x){return x*x;}
var f=function(x){return x*x;}
var ff= function fact(x){
if(x<=1)return 1;
else return x*fact(x-1);
}//把函数的引用存储到ff变量中,并未将函数引用存储到fact变量中,只是允许函数体用这个名字来引用自身。
f[0]=function(x){return x*x;};
a.sort(function(a,b){return a-b;});
var tensquared=(function(x){return x*x;})(10);//10*10=100,定义并调用函数
//可选参数
function copyPropertyNamesToArray(o,/*optional*/a){
if(!a)a=[];
//a=a||[];
for(var property in o)a.push(property);
return a;
}
var a=copyPropertyNamesToArray(o);
copyPropertyNamesToArray(p,a);
可变长度的参数列表对象:Argument对象
//Arguments对象允许完全地存储那些实际参数值,即使某些或全部参数名还没被命名。
function f(x,y,z){
if(arguments.length!=3){
throw new Error("function f called with "+arguments.length+
" arguments, but is expects 3 arguments");
}else{
console.log(arguments[0],arguments[1],arguments[2]);//对应x,y,z
return Math.sqrt(x*x+y*y+z*z);
}
}
//类似Math.max()求最大值
function max(/*……*/){
var m=Number.NEGATIVE_INFINITY;
for(var i=0;i<arguments.length;i++){
if(arguments[i]>m)m=arguments[i];
}
return m;
}
var largest=max(11,10,3,5,60,33,45,43);//60
function f(x){
print(x);//原始x值
arguments[0]=null;//通过arguments数组改变参数值
print(x);//null
}
//Arguments对象定义了callee属性,用来引用当前正执行的函数,可以用来允许对未命名函数递归地调用自身。
function(x){
if(x<=1)return 1;
return x*arguments.callee(x-1);
}
把对象属性用作参数
function arraycopy(/*array*/from,/*index*/ from_start,/*array*/to,/*index*/ to_start,/*integer*/length){
//code
}
function easycopy(args){ arraycopy(args.from,args.from_start||0,args.to,args.to_start||0,args.length);
}
var a=[1,2,3,4];
var b=new Array(4);
easycopy({from:a,to:b,length:4});
参数类型
function sum(a){//参数类型判断,如果非正确类型报错
if((a instanceof Array)||(a && typeof a=="object"&&"length" in a)){
var total=0;
for(var i=0;i<a.length;i++){
var element=a[i];
if(!element)continue;
if(typeof elment=="number")total+=element;
else {
throw new Error("sum():all array element must be numbers");
break;
}
}
return total;
}else{
throw new Error("sum():argument must be an number array");
}
}
function flexisum(a){//转换参数为合适的类型再计算
var total=0;
for(var i=0;i<arguments.length;i++){
var element=arguments[i];
if(!element)continue;
var n;
switch(typeof element){
case "number":
n=element;
break;
case "object":
if(element instanceof Array)
n=flexisum.apply(this,element);
else n=element.valueOf();
break;
case "function":
n=element();
break;
case "string":
n=parseFloat(element);
break;
case "boolean":
n=element?1:0;
break;
}
if(typeof n=="number"&&!isNaN(n))total+=n;
else throw new Error("sum():can't convert "+element+" to number");
}
return total;
}
作为数据的函数
function square(x){return x*x;}
var a=square(4);
var b=square;
var x=b(5);
var o=new Object();
o.square=function(x){return x*x;}
y=o.square(16);
var a=new Array();
a[0]=function(x){return x*x;};
a[1]=20;
a[2]=a[0](a[1]);
function add(x,y){return x+y;}
function subtract(x,y){return x-y;}
function multiply(x,y){return x*y;}
function divide(x,y){return x/y;}
function operation(operator,operand1,operand2){
return operator(operand1,operand2);
}
var i=operate(add,operate(add,2,3),operate(multiply,4,5));//(2+3)+(4*5)
var operators={
add:function(x,y){return x+y;},
subtract:(x,y){return x-y;},
multiply:(x,y){return x*y;},
divide:(x,y){return x/y;} ,
pow:Math.pow
}
function operate2(op_name,operand1,operand2){
if(typeof operators[op_name]=="function")
return operators[op_name](operand1,operand2);
else throw "unknown operator";
}
var j=operate2("add","hello",operate2("add"," ","world"));//"hello world"
var k=operate2("pow",10,2);
作为方法的函数
o.m=f;
o.m();
o.m(x,x+2);
var calculator={
operand1:1,
operand2:1,
compute:function(){
this.result=this.operand1+this.operand2;
}
};
calculator.compute();
print(calculator.result);//2
rect.setSize(width,height);
setRectSize(rect,width,height);
函数的属性和方法
//arguments.length传递的参数个数 Function.length函数声明的参数个数
function check(args){
var actual=args.length;
var expected=args.callee.length;
if(actual!=expected){
throw new Error("wrong number of arguments:expected:"+expected+";actually passed "+actual);
}
}
function f(x,y,z){
check(arguments);
return x+y+z;
}
//定义自己的函数属性
uniqueInteger.counter=0;//将属性存储在Function属性中
function uniqueInteger(){
return uniqueInteger.counter++;
}
//apply()和call()第一个参数都是要调用的函数对象,在函数体内这一参数是关键字this的值。call()的剩余参数是传递给调用的函数的值。
f.call(o,1,2);
//类似
o.m=f;
o.m(1,2);
delete o.m;
f.apply(o,[1,2]);
var biggest=Math.max.apply(null,array_of_numbers);
工具函数
function getPropertyNames(/*object*/o){
var r=[];
for(name in o)r.push(name);
return r;
}
function copyProperties(/*object*/ from,/*optional object*/to){
if(!to)to={};
for(p in from)to[p]=form[p];
return to;
}
function copyUnfefinedProperties(/*object*/ from,/*optional object*/to){
for(p in from){
if(!p in to)to[p]=form[p];
}
}
function filterArray(/*array*/a,/*boolean*/predicate){
var results={};
var length=a.length;
for(var i=0;i<length;i++){
var element=a[i];
if(predicate(element))results.push(element);
}
return results;
}
function mapArray(/*array*/a,/*function*/f){
var r=[];
var length=a.length;
for(var i =0;i<length;i++)r[i]=f(a[i]);
return r;
}
function bindMethod(/*object*/o,/*function*/f){
return function(){return f.apply(o,arguments);};
}
function bindArguments(/*function*/f/*,initial arguments...*/){
var boundArgs=arguments;
return function(){
var args={};
for(var i=1;i<boundArgs.length;i++)args.push(boundArgs[i]);
for(var i=0;i<arguments.length;i++)args.push(arguments[i]);
return f.apply(this.args);
}
}
作为闭包的嵌入函数
(function(){})();//未命名函数定义并调用
var x="global";
function f(){
var x="local";
function g(){console.log(x);}
g();
}
f();//local
function makeFun(x){
return function(){return x;};
}
var a=[makeFun(1),makeFun(2),makeFun(3)];
console.log(a[0]());
uniqueID=function(){
if(!arguments.callee.id)arguments.callee.id=0;
return arguments.callee.id++;
}
//存在问题:任何人都可以把uniqueID.id设置回0,并且违反函数所保证的它不会两次返回相同的值的约定。可以通过在只有该函数能够访问的闭包中存储持久性值的方法来防止这个月的问题:
uniqueID=(function(){
var id=0;
return function(){return id++;};
})();
//使用闭包的私有属性
function makeProperty(o,name,predicate){
var value;
o["get"+name]=function(){return value;}
o["set"+name]=function(v){
if(predicate&&!predicate(v)){
throw "set"+name+":invalid value "+v;
else
value=v;
}
}
}
var o={};
markeProperty(o,"Name",function(x){return typeof x=="string";});
o.setName("Frank");
console.log(o.getName());//Frank
o.setName(0);//"setName:invalid value 0"
断点是指函数中代码的执行停止下来,程序员获得一个机会查看变量的值、计算表达式、调用函数等的一个位置。Steve Yen所发明的断点工具,断电技术所使用一个闭包来捕获一个函数中的当前作用域(包括局部变量和函数的参数),并将它于全局的eval()函数组合起来,从而允许查看作用域。
var inspector=function($){return eval($);}//使用$减少在它想要检测的作用域内发生名字冲突的可能性。
//使用闭包的断点
function inspect(inspector,title){
var expression,result;
if("ignore" in arguments.callee) return;
while(true){
var message="";
if(title) message=title+"\n";
if(expression)message+="\n"+expression+" ==> "+result+"\n";
else expression="";
message+="Enter an expression to evaluate:";
expression =promot(message,expression);
if(!expression)return;
result=inspector(expression);
}
}
//使用断点计数计算阶乘
function factorial(n){
var inspector=function($){return eval($);}
inspect(inspector,"Entering factorial()");
var result=1;
while(n>1){
result=result*n;
n--;
inspect(inspector,"factorial() loop");
}
inspect(inspector,"Exiting factorial()");
return result;
}
Internet Explorer中的闭包和内存泄漏
IE浏览器对ActiveX对象和客户端的DOM原始使用垃圾收集的一种弱化形式。这些客户端的对象是引用计数的,并且当他们的引用数达到0的时候,就会被释放。当存在循环引用时候,这种方式失效,如,当一个核心JavaScript对象引用一个文档元素,并且稳定原始有一个属性(如一个事件处理器)指回到核心JavaScript对象的时候。
当闭包和IE的客户端变成一起使用时,这种循环闭包经常发生,。使用一个闭包时,闭包函数的调用对象包括所有函数参数和局部变量,所持续的时间都和闭包一样长。如果那些函数参数或局部变量的任意一个引用一个客户端对象,就可能会导致一次内存泄漏。
Function()构造函数
Function()构造函数期待任意数目的字符串参数,最后一个参数时函数的函数体。
注意:Function()构造函数并没有传递给任何一个参数来指定他所创建的函数名。和函数直接量一样,Function()构造函数创建了匿名的函数。
var f=new Function("x","y","return x*y;");
//等价于
function f(x,y){return x*y;}
var y="global";
function constructFunction(){
var y="local";
return new Funciton("return y");
}
console.log(constructFunction()());//global
//创建一个新的空对象
new Object();
var array=new Array(10);
var today=new Date();
//构造函数的工作是初始化一个新创建的对象,设置在使用对象前需要设置的所有属性。
function Rectangle(w,h){
this.width=w;
this.height=h;
}
var rect1=new Rectangle(2,4);//rect1={width:2,height:4}
var rect2=new Rectangle(8.5,11);//rect1={width:8.5,height:11}
//原型和继承
function computeAreaOfRectangle(r){
return r.width*r.height;
}
var r=new Rectangle(8.5,11);
r.area=function(){return this.width*this.height;}
var a=r.area();
function Rectangle(w,h){
this.width=w;
this.height=h;
this.area=function(){this.width*this.height;}
}
var r=new Rectangle(8.5,11);
var a=r.area();
function Rectangle(w,h){
this.width=w;
this.height=h;
}
Rectangle.prototype.area=function(){return this.width*this.height;}
var =new Rectangle(2,3);
r.hasOwnProperty("width");//true
r.hasOwnProperty("area");//false
"area" in r;//true
c.area()读取c的area属性 | → | area并不定义在c自身中,因此观察和c相关的原型对象的属性 | → | 这里才是area的定义,返回它,就好像它是c自身的属性一样 |
---|---|---|---|---|
A Circle object ,C{r=1.0,x=2.0,y=3.0} | → | ↓ | ||
c.pi=4;写入c的pi属性 | → | pi并不定义在c中,因此将其作为c的自身的新属性创建。 | ↓ | |
A Circle object ,C {r=1.0,x=2.0,y=3.0,pi=4} | 原型对象Circle.prototype{area=Circle_area,pi=3.14159} | |||
a=c.pi*c.r*c.r读取c的pi和r属性 | → | pi和r定义在c自身中,因此可以返回在那里找到的值,而不必再费劲查看原型对象 | ↑ | |
A Circle object ,d{r=2.1,x=0.0,y=0.0} | → | ↑ | ||
a=d.pi*d.r*d.r读取d的pi和r属性 | → | pi并不定义在d自身中,因此查看和d相关的原型对象和属性,r定义在d中,因此,返回这个值而不必查看原型 | → | 这里是pi的定义,返回它的值,就好像它真的是d的一个属性 |
扩展内建类型
String.prototype.endsWidth=function(c){
return (c==this.charAt(this.length-1));
}
var message="Hello World";
message.endsWidth("h");//false
message.endsWidth("d");//true
注意:不能为Object.prototype添加属性,所添加的任何属性和方法都可以用一个 for/in循环来枚举,将他们添加到Object.prototype就会使它们再每个单个的Javascript对象中都可见。一个空的对象{}应该没有枚举属性,任何添加到Object.prototype的内容都会变成这个空对象的一个可枚举属性,那么把对象用作关联数组的代码可能会产生问题。
//IE 4&5 无法使用Function.apply()
if(!Function.prototype.apply){
Function.prototype.apply=function(object,parameters){
var f=this;
var o=object||window;
var args=parameters||[];
o._$_apply_$_=f;
var argList=stringArgs.join(",");
var methodcall="o._$_apply_$_{"+argList+"};";
var result=eval(methodcall);
delete o._$_apply_$_;
return result;
}
}
//Firefox1.5实现新的数组方法 Array.map()
if(!Array.prototype.map){
Array.prototype.map=function(f,thisObject){
var results=[];
for(var len=this.length,i=0;i<len;i++){
result.push(f.call(thisObject,this[i],i,this));
}
return results;
}
}// Array.prototype.map
}//if(!Array.prototype.map)
实例方法
一个实例方法的实现使用this关键紫来引用调用它时所给予的所有对象或实例。一个实例方法可以针对类的任何实例来调用,但是,这并不意味者每个对象都包含该方法的一份自己的私有拷贝,这就是和对实例属性一样。相反,每个实例方法由一个类的所有实例来共享。再JavaScript中,通过构造函数的原型对象中一个属性设置为一个函数之,从而定义了一个实例方法。通过这种方法,构造函数所创造的所有对象都共享一个继承的执行该函数的引用,并且可以使用前面描述的方法调用语法来调用该函数。
在java和c++中,实例方法的作用域包括this对象。
在JavaScript中这些属性显式地指定this关键字。
每个类属性都只有一根拷贝,类属性本质上是全局的,类方法也是全局的。
类方法时通过一个构造函数来调用的,this关键字并不引用类的任何具体的实例,它引用的时构造函数自身。通常一个类的方法根本不使用this。
//通过构造函数自身的一个属性来模拟Javascript中的一个属性
Rectangle.UNIT=new Rectangle(1,1);
//Circle类
function Circle(radius){
this.r=radius;
}
Circle.PI=3.14159;
Circle.prototype.area=function(){return Circle.PI*this.r*this.r;}
Circle.max=function(a,b){
if(a.r>b.r)return a;
else return b;
}
var c=new Circle(1.0);//r=1.0
c.r=2.2;//r=2.2
var a=c.area();//a=15.205295600000001
var x=Math.exp(Circle.PI);//x=23.14063122695496
var d=new Circle(1.2);//r=1.2
var bigger=Circle.max(c,d);//c
//复数类
function Complex(real,imaginary){
this.x=real;
this.y=imaginary;
}
Complex.prototype.magnitude=function(){
return Math.sqrt(this.x*this.x+this.y*this.y);
}
Complex.prototype.negative=function(){
return Complex(-this.x,-this.y);
}
Complex.prototype.add=function(that){
return new Complex(this.x+that.x,this.y+that.y);
}
Complex.prototype.multiply=function(that){
return new Complex(this.x*that.x-this.y*that.y,this.x*that.y+this.y*that.x);
}
Complex.prototype.toString()+function(){
return "("+this.x+","+this.y+")";
}
Complex.prototype.equals=function(that){
return this.x==that.x&&this.y==that.y;
}
Complex.prototype.valueOf=function(){
return this.x;
}
Complex.sum=function(a,b){
return new Complex(a.x+b.x,a.y+b.y);
}
Complex.produce=function(a,b){
return new (a.x*bx-a.y*b*y,a.x*b.y+a.y*b.x);
}
Complex.ZERO=new Complex(0,0);
Complex.ONE=new Complex(1,0);
Complex.I=new Complex(0,1);
私有成员
数据封装,将属性变为私有的,并且通过专门的accessor方法才能读取和写入它们。
//不可变Rectangle对象,宽高不可变,只能通过accessor方法访问。
function ImmutableRectangle(w,h){
this.getWidth=function(){return w;};
this.getHeight=function(){return h;};
}
ImmutableRectangle.prototype.area=function(){
return this.getWidth()*this.getHeight();
}
通用对象模型
Circle.prototype.toString=function(){
return "[Circle of radius "+this.r+" ,centered at ("+this.x+","+this.y+").]";
}
var a=new Complex(5,4);
var b=new Complex(2,1);
var c=Complex.sum(a,b);// (7,5)
var d=a+b;//7
compareTo()应该接受一个单独的参数并且将这个参数和调用该方法所基于的对象进行比较。如果this对象小于参数对象,compareTo()应该返回一个小于0的值,如果this对象比参数对象大,该方法应该返回一个大于0 的值,如果两个对象相等,该帆帆应该返回0.
替代这个 | 使用这个 |
---|---|
a | a.compareTo(b)<0 |
a<=b | a.compareTo(b)<=0 |
a>b | a.compareTo(b)>0 |
a>=b | a.compareTo(b)>=0 |
a==b | a.compareTo(b)==0 |
a!=b | a.compareTo(b)!=0 |
Complex.prototype.compareTo=function(that){
if(!that||!that.magnitude||typeof that.magnitude !="function")
throw new Error("bad argument to Complex.compareTO()");
return this.magnitude()-that.magnitude();
}
//Array.sort()
complexNumbers.sort(new function(a,b){return a.compareTo(b)});
//另一种定义
Complex.compare=function(a,b){return a.compareTo(b);};
complexNumbers.sort(Complex.compare);
//1+0i和0+1i具有相同的模但实际值不相同
Complex.prototype.compareTo=function(that){
var result=this.x-that.x;
if(result==0)
return =this.y-that.y;
return result;
}
超类和子类
在JavaScript中,Object类是最通用的类,其他所有类都是专用化了的版本,或者说是Object的子类,另一种解释:Object是所有内建类的超类。所有类都从Object继承一些基本方法。
function Rectangle(w,h){
this.width=w;
this.height=h;
}
Rectangle.prototype.area=function(){return this.width*this.height;}
function PositionedRectangle(x,y,w,h){
Rectangle.call(this,w,h);//调用超类
this.x=x;
this.y=y;
}
PositionedRectangle.prototype=new Rectangle();
delete PositionedRectangle.prototype.width;
delete PositionedRectangle.prototype.height;
PositionedRectangle.prototype.constructor=PositionedRectangle;
PositionedRectangle.prototype.contains=function(x,y){
return (x>this.x&&x<this.x+this.width&&y>this.y&&y<this.y+this.height);
}
var r=new PositionedRectangle(2,2,2,2);
print(r.contains(3,3));//true
print(r.area());//4
print(r.x+","+r.y+","+r.width+","+r.height);//2,2,2,2
r instanceof PositionedRectangle //true
r instanceof Rectangle //true
r instanceof Object//true
//构造函数显式地调用超类的构造函数=》构造函数链
PositionedRectangle.prototype.superclass=Rectangle;
//等价
function PositionedRectangle(x,y,w,h){
this.superclass(w,h);
this.x=x;
this.y=y;
}
//调用被覆盖的方法
Rectangle.prototype.toString=function(){
return '['+this.width+','+this.height+']';
}
//超类的toString()的实现是超类的原型对象的一个属性,无法直接调用该方法,二十通过apply()调用该方法
PositionedRectangle.prototype.toString=function(){
return "{"+this.x+","+this.y+"}"+Rectangle.prototype.toString.apply(this);
}
PositionedRectangle.prototype.toString=function(){
return "{"+this.x+","+this.y+"}"+this.supperClass.prototype.toString.apply(this);
}
非继承的扩展
//从一个类借用方法供另一个类使用
function borrowMethods(borrowForm,addTo){
var from borrowFrom.prototype;
var to=addTo.prototype;
for(m in from){
if(type of from[m]!="function") continue;
to[m]=from[m];
}
}
//带通用的供借用的方法的混入类
function GenericToString(){}
GenericToString.prototype.toString=function(){
var props=[];
for(var name in this){
if(!this.hasOwnProperty(name))continue;
var value=this[name];
var s=name+":";
switch(typeof value){
case 'function':
s+="function";
break;
case 'object':
if(value instanceof Array)s+="array"
else s+=value.toString();
break;
default:
s+=String(value);
break;
}
props.push(s);
}
return "{"+props.join(",")+"}";
}
function GenericEquals(){}
GenericEquals.prototype.equals=function(that){
if(this==that) return true;
var propsInThat=0;
for(var name in that){
propsInThat++;
if(this[name]!==that[name])return false;
}
var propsThis=0;
for(name in this)propsInThis++;
if(propsInThis!=propsInThat)return false;
return true;
}
function Rectangle(x,y,w,h){
this.x=x;
this.y=y;
this.width=w;
this.height=h;
}
Rectangle.prototype.area=function(){return this.width*this.height;}
borrowMethods(GenericEquals,Rectangle);
borrowMethods(GenericToString,Rectangle);
function Colored(c){this.color=c;}
Colored.prototype.getColor=function(){return this.color;}
function ColoredRectangle(x,y,w,h,c){
this.superclass(x,y,w,h);
Colored.call(this,c);
}
ColoredRectangle.prototype=new Rectangle();
ColoredRectangle.prototype.constructor=ColoredRectangle;
ColoredRectangle.prototype.superclass=Rectangle;
borrowMethods(Colored,ColoredRectangle);
确定对象类型
typeof 从对象中区分出基本类型,任何函数类型都是function,任何数组的类型都是object
typeof null//object
typeof undefined//undefined
instanceof左侧是测试值,右侧是类的构造函数。注意,对象是它自己的类的一个实例,也是任何超类的一个实例。
o instanceof Object//true o为任何对象
typeof f=='function'//true
f instanceof Function//true
f instanceof Object //true
var d=new Date();
var isobj=d instanceof Object;//true
var realobj=d.constructor==Object;//false
instanceof 和constructor属性的缺点是它们只能允许根据已知的类来进行测试对象,无法检测位置的对象。
Object.toString(),toString()支队内建的对象类型有效,在Object.prototype中显式地引用默认函数,并使apply()在感兴趣的对象上调用它。
Object.prototype.toString.apply(o);
//扩展的typeof测试
function getType(x){
if(x==null) return "null";
var t=typeof x;
if(t!="object") return t;
var c=Object.prototype.toString.apply(x);//return "[object calss]"
c=c.substring(8,c.length-1);//"class"
if(c!="Object")return c;
if(x.constructor==Object) return c;
if("classname" in x.constructor.prototype&&
typeof x.constructor.prototype.classname=="string")
return "" ;
}
鸭子类型识别
//测试一个对象是否借用一个类的方法
function borrows(o,c){
if(o instanceof c)return true;
if(c==Array||c==Boolean||c==Date||c==Error||c==Function||c==Number||c==RegExp||c==String)
return undefined;
if(typeof o=="function") o=o.prototype;
var proto=c.prototype;
for(var p in proto){
if(type of proto[p]!="function") continue;
if(o[p]!=proto[p])return false;
}
return true;
}
//测试一个对象是否提供了方法
function provides(o,c){
if(o instanceof c)return true;
if(typeof o=="function")o=o.prototype;
if(c==Array||c==Boolean||c==Date||c==Error||c==Function||c==Number||c==RegExp||c==String)
return undefined;
var proto=c.prototype;
for(var p in proto){
if(typeof proto[p]!="function")continue;
if(!(p in o))return false;
if(typeof o[p]!="function")return false;
if(o[p].length!=proto[p].length)return false;
}
return true;
}
function Comparable(){};
Comparable.prototype.compareTo=function(that){
throw "Comparable.compareTo() is abstract. Don't invoke it!";
}
if(o.constructor==p.constructor&&provides(o,Comparable)){
var order=o.compareTo(p);
}
//测试类似数组的对象
function isArrayLike(x){
if(x instanceof Array)return true;
if(!("length" in x))return false;
if(typeof x.length!="number")return false;
if(x.length<0)return false;
if(x.length>0){
if(!((x.length-1) in x))return false;
}
return true;
}
//用来定义类的一个工具函数
function defineClass(data){
var classname=data.name;
var superclass=data.extend||Object;
var constructor=data.construct||function(){};
var methods=data.methods||{};
var statics=data.statics||{};
var borrows;
var provides;
if(!data.borrows)borrows=[];
else if(data.borrows instanceof Array)borrows=data.borrows;
else borrows=[data.borrows];
if(!data.provides)provides=[];
else if(data.provides instanceof Array)provides=dta.provides;
else provides=[data.provides];
var proto=new superclass();
for(var p in proto){
if(proto.hasOwnProperty(p))delete proto[p];
}
for(var i=0;i<borrows.length;i++){
var c=dta.borrows[i];
borrows[i]=c;
for(var p in c.prototype){
if(typeof c.prototype[p]!="function")continue;
proto[p]=c.prototype[p];
}
}
for(var p in methods)proto[p]=methods[p];
proto.constructor=constructor;
proto.superclass=superclass;
if(classname)proto.classname=classname;
for(var i=0;i<provides.length;i++){
var c=provides[i];
for(var p in c.prototype){
if(typeof c.prototype[p]!="function")continue;
if(p=="constructor"||p=="supperclass")continue;
if(p in proto&& typeof proto[p]=="function"&&proto[p].length==c.prototype[p].length)continue;
throw new Error("Class "+classname+" does not provide method "+c.classname+"."+p);
}
}
constructor.prototype=proto;
for(var p in statics)constructor[p]=data.statics[p];
return constructor;
}
//使用defineClass()方法
var Comparable=defineClass({
name:"Comparable",
methods:{CompareTo:function(that){throw "abstract";}}
});
var GenericEquals=defineClass({
name:"GenericEquals",
methods:{
equals:function(that){
if(this==that)return true;
var propsInThat=0;
for(var name in that){
propsInThat++;
if(this[name]!==that[name])return false;
}
var propsInThis=0;
for(name in this)propsInThis++;
if(propsInThis!=propsInThat)return false;
return true;
}
}
});
var Rectangle=defineClass({
name:"Rectangle",
construct:function(w,h){this.width=w;this.height=h;},
methods:{
area:function(){return this.width*this.height;},
compareTo:function(that){return this.area()-that.area();}
},
provides:Comparable
})
var PositionedRectangle=defineClass({
name:"PositionedRectangle",
extend:Rectangle,
construct:function(x,y,w,h){
this.superclass(w,h);
this.x=x;
this.y=y;
},
methods:{
isInside:function(x,y){
return x>this.x&&x<this.x+this.width&&y>this.y&&y<this.y+this.height;
}
},
statics:{
comparator:function(a,b){return a.compareTo(b);}
},
borrows:[GenericEquals]
})