故采用第二套解决方案,使用es5语法重写es6。
二、new.target 重写
es5的构造函数前面如果不用new调用,this指向window,对象的属性就得不到值了,所以之前都要在构造函数中通过判断this是否使用了new关键字来确保普通的函数调用方式都能让对象复制到属性。
1 function Person( uName ){
2 if ( this instanceof Person ) {
3 this.userName = uName;
4 }else {
5 return new Person( uName );
6 }
7 }
8 Person.prototype.showUserName = function(){
9 return this.userName;
10 }
11 console.log( Person( 'ghostwu' ).showUserName() );
12 console.log( new Person( 'ghostwu' ).showUserName() );
在es6中,为了识别函数调用时,是否使用了new关键字,引入了一个新的属性new.target:
- 如果函数使用了
new,那么new.target就是构造函数; - 如果函数没有使用
new,那么new.target就是undefined; es6的类方法中,在调用时候,使用new,new.target指向类本身,没有使用new就是undefined;
1 function Person( uName ){
2 if( new.target !== undefined ){
3 this.userName = uName;
4 }else {
5 throw new Error( '必须用new实例化' );
6 }
7 }
8 // Person( 'ghostwu' ); //报错
9 console.log( new Person( 'ghostwu' ).userName ); //ghostwu
使用new之后,new.target就是Person这个构造函数,那么上例也可以用下面这种写法:
1 function Person( uName ){
2 if ( new.target === Person ) {
3 this.userName = uName;
4 }else {
5 throw new Error( '必须用new实例化' );
6 }
7 }
8
9 // Person( 'ghostwu' ); //报错
10 console.log( new Person( 'ghostwu' ).userName ); //ghostwu
1 class Person{
2 constructor( uName ){
3 if ( new.target === Person ) {
4 this.userName = uName;
5 }else {
6 throw new Error( '必须要用new关键字' );
7 }
8 }
9 }
10
11 // Person( 'ghostwu' ); //报错
12 console.log( new Person( 'ghostwu' ).userName ); //ghostwu
上例,在使用new的时候, new.target等于Person。
掌握new.target之后,接下来,我们用es5语法改写上文中es6的类语法。
1 let Person = ( function(){
2 'use strict';
3 const Person = function( uName ){
4 if ( new.target !== undefined ){
5 this.userName = uName;
6 }else {
7 throw new Error( '必须使用new关键字' );
8 }
9 }
10
11 Object.defineProperty( Person.prototype, 'sayName', {
12 value : function(){
13 if ( typeof new.target !== 'undefined' ) {
14 throw new Error( '类里面的方法不能使用new关键字' );
15 }
### 最后
全网独播-价值千万金融项目前端架构实战

从两道网易面试题-分析JavaScript底层机制

RESTful架构在Nodejs下的最佳实践
**[开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】](https://bbs.youkuaiyun.com/topics/618166371)**

一线互联网企业如何初始化项目-做一个自己的vue-cli

思维无价,看我用Nodejs实现MVC

代码优雅的秘诀-用观察者模式深度解耦模块

前端高级实战,如何封装属于自己的JS库

VUE组件库级组件封装-高复用弹窗组件

本文介绍了JavaScript中new.target在ES5和ES6构造函数以及类中的作用,展示了如何使用new.target来判断函数是否通过new调用,并提供了从ES6类语法转换为ES5的示例。
1019

被折叠的 条评论
为什么被折叠?



