手写:javascript中的关键字new
简单介绍一下new
new再熟悉不过了,new后面跟着构造函数,可以创建对象,这个对象的原型指向构造函数的原型对象,说起来可能有点绕,直接看代码吧
function Person(name, age){
    this.name = name;
    this.age = age;
}
let person1 = new Person("张三", 22);
console.log(person1.__proto__ === Person.prototype); // true
console.log(person1 instanceof Person); // true
而new后面的构造函数,也可以这样调用
let person2 = new Person;
其实这样写就等同于,不传任何参数调用构造函数
let person2 = new Person();
打印person2可以看到
Person { name: undefined, age: undefined }
一般情况下,构造函数是不定义返回值的,那我们试一试,如果构造函数定义返回值,会怎么样,首先,返回一个字符串
function Person(name, age){
    this.name = name;
    this.age = age;
    return "person"
}
let person1 = new Person("张三", 22);
let person2 = new Person;
console.log(person1); // Person { name: '张三', age: 22 }
console.log(person2); // Person { name: undefined, age: undefined }
console.log(person1.__proto__ === Person.prototype); // true
console.log(person1 instanceof Person) // true
看来构造函数返回字符串没有影响,那返回一个对象呢
function Person(name, age){
    this.name = name;
    this.age = age;
    return {
        id : "person"
    }
}
let person1 = new Person("张三", 22);
let person2 = new Person;
console.log(person1); // { id: 'person' }
console.log(person2); // { id: 'person' }
console.log(person1.__proto__ === Person.prototype); // false
console.log(person1 instanceof Person) // false
可以看到,如果返回一个对象,那么通过new关键字创建的对象则为构造返回的值,且也不继承于构造的原型对象
手写
了解的差不多了,开始手写
首先定义一个函数名为myNew,参数接收一个function,也就是构造函数,
function myNew(Constructor){
    if(typeof Constructor !== "function"){
        throw "参数Constructor必须是方法(function)"
    }
}
ok,紧接着需要创建一个新的对象,并且这个对象的原型指向的是构造函数的原型对象
所以
function myNew(Constructor){
    if(typeof Constructor !== "function"){
        throw "参数Constructor必须是方法(function)"
    }
    let obj = {};
    obj.__proto__ = Constructor.prototype;
}
接着,因为执行构造函数的是,需要传参数,并且在构造函数内部会为this赋值,所以我们需要执行构造函数,并且把构造函数内的上下文指向到obj
function myNew(Constructor){
    if(typeof Constructor !== "function"){
        throw "参数Constructor必须是方法(function)"
    }
    let obj = {};
    obj.__proto__ = Constructor.prototype;
    // 获取参数
    let params = Array.from(arguments);
    // 把参数第一位,也就是构造函数移除
    params.shift();
    // 执行构造函数,并且吧上下文只设置为obj
    Constructor.apply(obj, params);
    return obj
}
但是我们需要前面提到了,有时候构造函数是可能有返回值的,所以最终代码为
最终源码
function myNew(Constructor){
    if(typeof Constructor !== "function"){
        throw "参数Constructor必须是方法(function)"
    }
    let obj = {};
    obj.__proto__ = Constructor.prototype;
    // 获取参数
    let params = Array.from(arguments);
    // 把参数第一位,也就是构造函数移除
    params.shift();
    // 执行构造函数,并且吧上下文只设置为obj
    let result = Constructor.apply(obj, params);
    // 判断构造函数执行返回是不是object,是的话返回这个对象
    return typeof result === "object" ? result : obj
}
我们以之前的例子来验证一下
function Person(name, age){
    this.name = name;
    this.age = age;
    return "person"
}
let person1 = myNew(Person, "张三", 22);
let person2 = myNew(Person);
console.log(person1); // Person { name: '张三', age: 22 }
console.log(person2); // Person { name: undefined, age: undefined }
console.log(person1.__proto__ === Person.prototype); // true
console.log(person1 instanceof Person) // true
构造有返回值的时候
function Person(name, age){
    this.name = name;
    this.age = age;
    return {
        id : "person"
    }
}
let person1 = myNew(Person, "张三", 22);
let person2 = myNew(Person);
console.log(person1); // { id: 'person' }
console.log(person2); // { id: 'person' }
console.log(person1.__proto__ === Person.prototype); // false
console.log(person1 instanceof Person) // false
与前面完全一致
手写:javascript中的关键字new的更多相关文章
- 深入解析Javascript中this关键字的使用
		深入解析Javascript中面向对象编程中的this关键字 在Javascript中this关键字代表函数运行时,自动生成的一个内部对象,只能在函数内部使用.比如: function TestFun ... 
- JavaScript中this关键字的使用比较
		JavaScript中this关键字的使用比较 this关键字在JavaScript中,用的不能说比较多,而是非常多.那么熟悉this关键字的各种用法则显得非常关键. this有时候就是我们经常说的上 ... 
- 大前端学习笔记整理【五】关于JavaScript中的关键字——this
		写在前面 工作有那么一段时间了,但是在工作中,发现自己的理论知识还是有所欠缺.特别是在javascript上,很多东西其实自己属于知道要用这个,但是不知道为什么要这么用...这种情况很是尴尬了,所以写 ... 
- javascript中this关键字详解
		不管学习什么知识,习惯于把自己所学习的知识列成一个list,会有助于我们理清思路,是一个很好的学习方法.强烈推荐. 以下篇幅有点长,希望读者耐心阅读. 以下内容会分为如下部分: 1.涵义 1.1:th ... 
- JavaScript中var关键字的使用详解
		作用 声明作用:如声明个变量. 语法 ? 1 var c = 1; 省略var 在javascript中,若省略var关键字而直接赋值,那么这个变量为全局变量,哪怕是在function里定义的. ? ... 
- [No000069]Javascript中this关键字详解
		Quiz 请看下面的代码,最后alert出来的是什么呢?(chrome下按F12,选择Console直接复制粘贴运行) var name = "Bob"; var nameObj ... 
- javascript中的关键字和保留字
		javascript中关键字的问题,将名称替换了下,确实就没有问题了.现在将它的关键字和保留字贴出来,便于日后查看和避免在次出现类似的问题. 1 关键字breakcasecatchcontinuede ... 
- TensorFlow实现Softmax Regression识别手写数字中"TimeoutError: [WinError 10060] 由于连接方在一段时间后没有正确答复或连接的主机没有反应,连接尝试失败”问题
		出现问题: 在使用TensorFlow实现MNIST手写数字识别时,出现"TimeoutError: [WinError 10060] 由于连接方在一段时间后没有正确答复或连接的主机没有反应 ... 
- javascript中new关键字详解
		和其他高级语言一样 javascript 中也有 new 运算符,我们知道 new 运算符是用来实例化一个类,从而在内存中分配一个实例对象. 但在 javascript 中,万物皆对象,为什么还要通过 ... 
随机推荐
- 结对项目:四则运算(C语言)
			github地址:https://github.com/nilonger/arithmetic 结对伙伴:杨锐龙+黄海钊 一.项目要求 1.1 题目:实现一个自动生成小学四则运算题目的命令行程序(也可 ... 
- IdentityServer4网页(单点)登陆入门
			前言 本篇说说ids中的网页登陆以及单点登陆的大致原理,主要是以基本跑通为目的,下一篇开始会详细说明集成ids网页登陆原理. 最好先熟悉以下知识: asp.net core asp.net core的 ... 
- 将Darknet编译成API形式的so文件
			Makefile: GPU= CUDNN= OPENCV= OPENMP= DEBUG= ARCH= -gencode arch=compute_30,code=sm_30 \ -gencode ar ... 
- 实用教程!SPSSAU验证性因子分析思路总结
			验证性因子分析,是用于测量因子与测量项(量表题项)之间的对应关系是否与研究者预测保持一致的一种研究方法.尽管因子分析适合任何学科使用,但以社会科学居多. 目前有很多软件都可以非常便利地实现验证性因子分 ... 
- 《Head First 设计模式》:外观模式
			正文 一.定义 外观模式提供了一个统一的接口,用来访问子系统中的一群接口.外观定义了一个高层接口,让子系统更容易使用. 要点: 外观模式将一个或数个类的复杂的一切都隐藏在背后,只显露出一个干净美好的外 ... 
- Qt QTranslator  实现多语言转换
			1.在*.pro文件里面添加TRANSLATIONS += English.tsChinese.ts根据自己想要添加多少种语言和什么语言视情况添加. 2.a.通过qt ui设计师添加的文字 首先记得选 ... 
- JavaScript函数的高级用法
			1.函数的定义和调用 1.1函数的定义方式 方式1 函数声明方式 function 关键字 (命名函数) function fn(){} 方式2 函数表达式(匿名函数) var fn = functi ... 
- rust 模块组织结构
			rust有自己的规则和约定用来组织模块,比如一个包最多可以有一个库crate,任意多个二进制crate.导入文件夹内的模块的两种约定方式... 知道这些约定,就可以快速了解rust的模块系统. 先把一 ... 
- java调用C# webService发布的接口
			java调用C# webService发布的接口 java调用C# webService方式有很多种我这里只介绍一种 首先需要引入axis的jar包 axis的maven坐标如下 <depend ... 
- Codeforces1393 题解(A-D)
			AC代码 A. Rainbow Dash, Fluttershy and Chess Coloring 可以推导出\(f_1 = 1, f_2 = 2, ..., f_n = f_{n - 2} + ... 
