[技术学习]js接口继承
js是面向对象语言,但是js又缺乏了面向对象的诸多特性,比如继承,没有接口继承也没有父类继承,因此有时候需要人工来实现继承。
一、首先看下java中面向对象的继承:
//定义类鸟类的飞行动作 interface BirdFlyable { public void fly(); } //鸟类 class Bird implements BirdFlyable{ public void fly(){ System.out.println("bird is flying"); } } //蝙蝠同样可以飞行,不过没有继承鸟类的动作 class Bianfu{ public void fly(){ System.out.println("Bianfu is flying"); } } public class Test{ //定义fly方法,要求传入鸟类飞行方法 public void fly(BirdFlyable flyable){ flyable.fly() } public static void main(String[] args) { Bird bird=new Bird(); fly(bird);//bird is flying //Bianfu bianfu=new Bianfu(); //fly(bianfu) //执行蝙蝠方法,报错 } }
此处可以看到fly(bird)执行结果为bird is flying,如果Bird继承了BirdFlyable却没有实现fly方法,会在编译时候报错,相反,如果使用fly(bianfu),因为蝙蝠没有继承接口,同样也会报错,因此java中的接口继承为严格的继承。
二、python继承
#定义鸟类飞行方法 class BirdFlyable(): def fly(self): pass #定义鸟类 class Bird(BirdFlyable): def fly(self): print('bird is flying') #定义蝙蝠 class Bianfu(): def fly(self): print('Bianfu is flying') #增加方法,希望传入鸟类飞行方法 def fly1(birdFlyable): birdFlyable.fly(); #事实证明,传入蝙蝠也是可以的 fly1(Bird())#bird is flying fly1(Bianfu())#bird is flying
在python中因为方法中的参数没有明确规定类型,因此只要执行时候含有此方法,即可完成功能。
这就是动态语言的“鸭子类型”,它并不要求严格的继承体系,一个对象只要“看起来像鸭子,走起路来像鸭子”,那它就可以被看做是鸭子。
三、js中实现接口
目前js中实现接口有三种方法:
1、文档注释,即预先在文档注释中写好接口,规定接口中的方法,开发人员根据注释编写接口,此方法其实用处不大,无法强制开发人员继承接口。
2、属性检测法:在定义类的时候,规定类继承了哪几个接口,此方法只能检测接口名称,不能检测具体实现方法。
3、鸭式辩型法:与python中一样,来源与“看起来像鸭子,走起路来像鸭子”,那它就可以被看做是鸭子的理论,重点关注类实现了哪些方法,而忽略继承了那个接口,从而判断类的继承,实现方法如下:
//鸭式辩型法 实现javascript接口最完美的方法 //核心方法是检测是否实现了接口的每一个方法 //面向对象,代码实现统一,也解耦了 //1、接口类 /** *接口需要两个参数,1、名字,2、抽象方法名称数组 */ var Interface=function(name,methods){ //如果参数大于一个,且第二个参数为数组,否则抛出异常 if(arguments.length>=2&&(arguments[1] instanceof Array)){ this.name=name; this.methods=[]; for(var index in methods){ if(typeof methods[index]=='string'){ this.methods.push(methods[index]); }else{ throw new Error('方法名应该是字符串') } } }else{ throw new Error("接口需要两个参数,1、名字,2、抽象方法名称数组"); } } //定义两个接口,分别为增删改查 var interface1=new Interface('interface1',['add','delete']); var interface2=new Interface('interface2',['modify','query']); /** *定义接口检验类,主要检验是否实现了接口中的所有方法 *接口中传递参数为:1、需要检测的类,2、其他接口参数 */ Interface.checkInterface=function(object){ if(arguments.length>=2){ for(var i=1,len=arguments.length;i<len;i++){ //获取需要检测的接口 var interface=arguments[i]; //获取接口中所含的所有方法 for(var j=0,methods=interface.methods;j<methods.length;j++){ var method=methods[j]; if(!object[method]||(typeof object[method]!='function')){ throw new Error("方法"+method+"没有实现"); } } } }else{ throw new Error('接口检验错误,参数至少两位') } } //定义From类 //实现接口1和接口2中的方法 function Form(){ this.add=function(){ alert('add'); }; this.delete=function(){ alert('delete') }; this.modify=function(){ alert('modify') }; this.query=function(){ alert('query') } } //实例化类 var form =new Form(); //做借口检查 Interface.checkInterface(form,interface1,interface2) form.add();