Js的call和apply以及ES5、ES6的继承


call方法


MDN关于call方法的解释

call() 方法调用一个函数, 其具有一个指定的this值和分别地提供的参数(参数的列表)。

注意:该方法的作用和 apply() 方法类似,只有一个区别,就是call()方法接受的是若干个参数的列表,而apply()方法接受的是一个包含多个参数的数组


例子:

function Person(name,age){
    this.name=name
    this.age=age
}


function Student(name,age){
    Person.call(this,name,age)
    this.grade="高三"
}


var s1=new Student("Andy",18)

console.log(s1.name)  // Andy
console.log(s1.age)   // 18

console.log(Person.prototype===Student.prototype)    //  false
console.log(Person.prototype.constructor===Student.prototype.constructor)    //  false



语法

fun.call(thisArg, arg1, arg2, ...)

参数

  thisArg

fun函数运行时指定的this值。

需要注意的是,指定的this值并不一定是该函数执行时真正的this值,如果这个函数处于非严格模式下,则指定为nullundefinedthis值会自动指向全局对象(浏览器中就是window对象),同时值为原始值(数字,字符串,布尔值)的this会指向该原始值的自动包装对象。

arg1, arg2, ...

指定的参数列表

返回值

返回值是你调用的方法的返回值,若该方法没有返回值,则返回undefined。


描述

可以让call()中的对象调用当前对象所拥有的function。你可以使用call()来实现继承:写一个方法,然后让另外一个新的对象来继承它(而不是在新对象中再写一次这个方法)。


现在再来看一下上面的那个例子


function Person(name,age){
    this.name=name
    this.age=age
}



function Student(name,age){
    Person.call(this,name,age);
    this.grade="高三";
}


var s1=new Student("Andy",18)

console.log(s1.name)  // Andy
console.log(s1.age)   // 18

在这个例子中,函数Person里面有this,这个this 指向的是什么要看你是怎么调用的,

如果你直接Person()直接这样调用,那么这个this指向的就是全局对象window

如果说你作为函数Person new出来的一个实例来调用,那么这个this指向的就是这个实例对象。

上面代码中,s1Student new出来的一个实例,那么这个实例在使用的时候,Student内部的this指向的就是这个实例对象.

那么Student这个函数里面的this指向的就是实例对象s1Person.call(this,name,age)是一个立即执行的函数,意思是说执行这个Person函数,并且这个函数内部的this指向s1这个实例对象。

最上面的function Person(){}这个函数内部的this指向的就是s1,结合 this.name=name; this.age=age就可以看出s1的name属性就是参数name,age属性就是参数age

所以打印s1.names1.age都能打印出来。

另外,特别需要注意的是, 使用关键字 new用来new一个实例的时候,会自动调用对象方法。 就是说在本例中,new Student("Andy",18) 的时候,会去调用 Student这个方法!!!


Student方法中call了一下,并且传入了和Person方法中同样的参数 nameage是让Student这个方法new出来的实例中都有了nameage这两个属性。

事实上,当你尝试打印 Person.hasOwnProperty('age') Student.hasOwnProperty('age')结果都会是false

console.log(Person.hasOwnProperty('age'))   // false

console.log(Student.hasOwnProperty('age'))  // false

console.log(s1.hasOwnProperty('age'))      // true

var p1=new Person('小明',5)
console.log(p1.hasOwnProperty('age'))      // true

说明nameage这两个属性都是实例对象上的。



Apply方法

apply方法的作用和call方法一样。 只是传参的方式不一样,上面的call方法可以看到nameage属性实际上是一个一个传进去的。 而apply方法是可以以数组的形式一下子全放进去。

上面的例子换成apply就是下面的这种写法


function Person(name,age){
    this.name=name
    this.age=age
}



function Student(name,age){
    Person.apply(this,[name,age]);
    this.grade="高三";
}


var s1=new Student("Andy",18)

console.log(s1.name)  // Andy
console.log(s1.age)   // 18



es5的继承

// es5的继承

function Person(name,age){
  this.name=name
  this.age=age
}



function Student(name,age,grade){
  Person.call(this,name,age);
  this.grade=grade;
}

// Object.create() 方法是用来创建一个新的对象,该对象的原型指向括号里面的参数
Student.prototype=Object.create(Person.prototype)
// 上面的这条语句是说,创建一个新的对象,该对象的原型指向Person.prototype,
并将它赋值给Student.prototype

// 再设置 constructor属性等于Student
Student.prototype.constructor=Student


//  以上就是es5的继承


//   需要注意的是,子类的原型上是没有父类的属性和方法的

console.log(Student.prototype.hasOwnProperty("age")) 
//  false 子类的原型上是没有父类的属性和方法的

// 子类的实例上才有这些属性和方法
var s1=new Student('张三',30)
console.log(s1.hasOwnProperty("age"))  //  true

ES5中这种最简单的继承,实质上就是将子类的原型设置为父类的实例。

需要经过
    1.定义祖先
    2.定义祖先可继承的变量/方法
    3.定义继承的类(构造函数),并在类中调用组件的方法 call和apply方法
    4.使用 prototyoe定义继承关系
    5.重新将constructor指向自己



1. 为什么在子类上需要call一下?

   :如果不call,用子类new出来的实例上就不会有父类的属性和方法。

实际上nameage这两个属性,或者说父类上的方法,你call了之后,等于说是在子类的实例上有了父类的属性和方法,子类本身是没有这些个属性和方法的。 不信你call了之后打印 Student.prototype.hasOwnProperty("age") 一定是 false的!

2. 为什么需要Object.create()

   :新创建对象的原型对象

3. 为什么需要Student.prototype.constructor=Student

   :重新将constructor指向自己

ES5 的继承,实质是先创造子类的实例对象this,然后再将父类的方法添加到this上面(Parent.call(this))。

上面这句话就很好理解了。


es6的继承

//  es6的继承

//定义类

class Person {
    // constructor方法,就是构造方法
    // 如果没有定义构造方法,js会自动为其添加
    //  constructor方法默认返回实例对象(即this),完全可以指定返回另外一个对象 
   // 例如: return Object.create({name:"John",age:31})
    //  new一个类的实例的时候,会立即调用constructor方法
    constructor(name){
        this.name=name
        this.showName=function(){
            console.log(this.name)
        }
    }

    //  下面的这个sayHello方法它是不可枚举的,这一点与es5不同
    //  需要注意的是,下面的这个sayHello方法不是定义在类Person上的,也不是定义在类的实例对象上的,
    // 而是定义在类Person的原型上的
    //  Person.hasOwnProperty('sayHello')  // false
    //  p1.hasOwnProperty('sayHello')  // false
    //  Person.prototype.hasOwnProperty('sayHello')  // true
    //  p1.__proto__hasOwnProperty('sayHello')  // true
    sayHello(){
            console.log('hello')
    }
}

var p1=new Person("xiao8")

console.log(Object.keys(p1))    
//  ["name", "showName"]  看到没有,sayHello不可枚举,constructor内部定义的变量与方法可枚举


//构造函数的prototype属性,在 ES6 的“类”上面继续存在。   
//事实上,类的所有方法都定义在类的prototype属性上面。
console.log(Person===Person.prototype.constructor)   //  true





//实现继承

class Student extends Person{
    constructor(name,grade){
        //调用父类的构造函数,用来新建父类的this对象
        //super作为函数调用时,返回的是子类B的实例,super内部的this指向B
        //super相当于 A.prototype.constructor.call(this)
        //super作为函数只能用在constructor中
        super(name)
        this.grade=grade
    }

    toString() {
        //super作为对象使用时,指向父类的原型对象。
        //在静态方法中指向父类
        //定义在父类实例上的方法是没办法用的
        return this.name + ' ' +super.sayHello();//调用父类的方法
    }
}


//可以使用getPrototypeOf方法来获取父类
console.log(Object.getPrototypeOf(Student)===Person)   // true

//这里和es5不一样
//对象有属性__proto__,指向该对象的构造函数的原型对象。
//方法除了有属性__proto__,还有属性prototype,prototype指向该方法的原型对象。

var p1 = new Person("Andy");
var s2 = new Student("小华",'三年级');

console.log(p1.__proto__ === p1.__proto__)// true
console.log(s2.__proto__.__proto__ === p1.__proto__) // true

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,701评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,649评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,037评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,994评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,018评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,796评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,481评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,370评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,868评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,014评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,153评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,832评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,494评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,039评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,156评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,437评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,131评论 2 356

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,645评论 18 399
  • Scala与Java的关系 Scala与Java的关系是非常紧密的!! 因为Scala是基于Java虚拟机,也就是...
    灯火gg阅读 3,448评论 1 24
  • 大家好,我是IT修真院萌新分院第3期的学员张晓琳,一枚正直、纯洁、善良的前端程序员今天给大家分享一下,修真院官网j...
    Demon_0481阅读 567评论 0 2
  • 这周主题下的内容狠狠的撞进了我的内心,让我更加清楚的看到了自己。 原来剖析自己,看见真实的自己,了解自己,虽然痛苦...
    遇见橙子阅读 124评论 0 0
  • 1.肩膀的三角肌是把肩膀变宽的肌肉之一,分为前束后束和中束。介绍一下联系三角肌前束的动作,其动作要领是:双脚分开与...
    青木川_阅读 847评论 0 0