通过例子深入理解javascript中的new操作符

1.首先看一道题目

1 function Cat(name,age){
2      this.name=name;
3      this.age=age;
4 }
5 console.log(new Cat('miaomiao',18));  
6 //Cat {name: "miaomiao", age: 18}

2.那么这里面的this指的是什么呢

1 function Cat(name,age){
2       console.log(this);//Cat {}
3       this.name=name;
4       this.age=age;
5 }
6 new Cat('miaomiao',18);

3.我们发现this是一个名为Cat的空对象,那么后两句(this.name=name;this.age=age)就相当于var Cat={};Cat.name=name;Cat.age=age;是这样的么我们来试一下

1 function Cat(name,age){
2       var Cat = {};
3       Cat.name=name;
4       Cat.age=age;
5 }
6 console.log(new Cat('miaomiao',18));  
7 //Cat {}

4.发现并不是那么回事,这是为什么,在javascript中如果没有return 那么函数就会默认return this为了验证我们在函数最后面return Cat即可

1 function Cat(name,age){
2       var Cat = {};
3       Cat.name=name;
4       Cat.age=age;
5       return Cat;
6 }
7 console.log(new Cat('miaomiao',18));  
8 //Object {name: "miaomiao", age: 18}

5.好像成功了,我们和之前对比一下

1 function Cat(name,age){
2       this.name=name;
3       this.age=age;
4  }
5  console.log(new Cat('miaomiao',18));  
6 //Cat {name: "miaomiao", age: 18}

6.函数的作用我们理解的差不多,下面开始探索NEW,试想如果我们不返回对象而是返回null或者其他类型的会是什么效果呢

1 function Cat(name,age){
2        var Cat = {};
3        Cat.name=name;
4        Cat.age=age;
5        return undefined//或者null 123 '123'等非对象;
6 }
7 console.log(new Cat('miaomiao',18));//Cat{}

7.其他的一律输出空对象

那么我们试着写一个类似new的函数功能

 1 function Cat(name,age){
 2    this.name=name;
 3    this.age=age;
 4 }
 5 function New(){
 6     var obj={};
 7     var res=Cat.apply(obj,arguments);
 8     return typeof res==='object'?res:obj
 9 }
10 console.log(New('mimi',18))
11 //Object {name: "mimi", age: 18}

8.这样就大功告成了么  当然不是,我们接着往下看

 1  function Cat(name,age){
 2      this.name=name;
 3      this.age=age;
 4   }
 5   Cat.prototype.sayHi=function(){
 6        console.log('hi')
 7   }
 8   function New(){
 9       var obj={};
10       var res=Cat.apply(obj,arguments);
11       return typeof res==='object'?res:obj
12   }
13 
14  console.log(new Cat('mimi',18).sayHi())
15  console.log('-------------');
16 console.log(New('mimi',18).sayHi());
17 
18 VM841:7 hi
19 VM841:15 undefined
20 VM841:16 -------------
21 VM841:17 Uncaught TypeError: New(...).sayHi is not a function(…)

9.红色的事报错,说明new操作不仅仅关注函数的本身 还关心他的原型 prototype

上面分析了那么多,现在进入正题。

  普通对象是没有prototype属性的,只有隐藏属性__proto__(IE上也有该隐藏属性,但是使用obj.__proto__不能输出东西,所以建议不要使用__proto__属性)。而函数对象则两者兼有。prototype属性指向的是函数对象的原型对象,对象的__proto__属性是创建实例对象的时候对应的函数对象的原型对象。

 1  function Cat(name,age){
 2      this.name=name;
 3      this.age=age;
 4   }
 5   Cat.prototype.sayHi=function(){
 6        console.log('hi')
 7   }
 8   function New(){
 9       var obj={};
10       obj.__proto__=Cat.prototype;
11       var res=Cat.apply(obj,arguments);
12       return typeof res==='object'?res:obj
13   }
14 
15  console.log(new Cat('mimi',18).sayHi())
16  console.log('-------------');console.log(New('mimi',18).sayHi());
17 
18 VM844:7 hi
19 VM844:16 undefined
20 VM844:17 -------------
21 VM844:7 hi
22 VM844:17 undefined
23 undefined

10.ok大功告成

posted on 2017-08-19 01:16  -韩帅  阅读(1299)  评论(0编辑  收藏  举报

导航