Javascript 中 call 的两种用法

用法一(常见用法):

表现形式为:一个对象.方法.call(另一个对象),意义是用另一个对象代替当前对象,执行当前对象的方法。先看示例:

 function Class1(){
this.name = "class1";
this.showName = function(){
alert(this.name);
}
}
function Class2(){
this.name = "class2";
}
var c1 = new Class1();
var c2 = new Class2(); c1.showName.call(c2);
c2.showName(); //不能执行

我们先分别定义了两个类,分别是Class1和Class2,它们的主要区别是Class2比Class1多了一个showName()方法。接着定义了与Class1、Class2对应的对象c1和c2,但是奇迹出现了,当我们执行 c1.shoName.call(c2) 时,会弹出c2的name值,这是为什么?其实,我们执行的仍然是c1的方法,只不过临时起意地把对象c2换成对象c1,执行完毕后c2没有因此多了些什么方法。为了检测c2有没有多了方法,示例加了行: c2.showNmae(); 它是不能被执行的,浏览器会报出 Object #<Class2> has no method 'showName' 的错误。

为什么要这么做?前面说了,这是一种临时起意的使用方法,我们就是利用其高效的编程而已。但这并不是没有限制的,假设用c1和c2表示被替换的对象和替换对象,用fun1表示c1固有的方法。1、当fun1不需要参数,且不用到父函数中的任意局部变量时,其实c1.fun1.call(c2) 和 c1.fun1() 没什么区别;2、当fun1不需要参数但是用到父类中的变量,那么就要求这个变量在c1和c2的父类中都存在;3、当fun1需要参数时,形式要改写成c1.fun1.call(c2, 参数1, 参数2, ...参数n),这个时候生成c1的function中的变量名就不必和生成c2的function的变量名同名了,只需对应即可。其实,我们在用到call的这种用法时,c2和c1往往在结构和功能上已经有很大的相似性了,所以上述三点很容易避免。

用法二:

在function的定义过程中使用,表现形式:另一个已存在函数.call(this),它可以改变this的指向,实现一种类似于继承的功能。看个例子:

 function Animal(name){
this.name = name;
this.showName = function(){
alert(this.name);
}
}; var animal = new Animal("small_animal");
animal.showName(); //alert("small_animal") function Cat(name){
Animal.call(this, name);
}; //var Animal = null; //取消注释试一试 var cat = new Cat("black_cat");
cat.showName(); //alert("black_cat")

先不看注释行,我们的程序做了四件事情:第一步,定义一个名为 Animal 的类,并且其包含 showName() 方法;第二步,基于 Animal 生成一个对象 animal,看看它的 showName() 方法能不能执行;第三步,定义一个名为 Cat 的 function,用到了 call 函数;第四步,基于 Cat 生成对象 cat,看看它的 showName() 方法能不能执行。结果表明两次测试都成功了。那第三步做了什么,这是不是实现了继承?

上一篇:关于input内容监听(change)


下一篇:ceph存储之查找对象