当前位置: 首页 > news >正文

JS创建对象模式及其对象原型链探究(一):Object模式

Object模式

创建一个Object实例,再为其添加属性和方法。

这是创建自定义对象最简单的方式。

1.创建对象

// 创建person对象
var person = new Object();
person.name = "Mike";
person.age = 20;
person.job = "student";
person.showName = function(){    
    console.log("this.name = " + this.name);
};

person.consThis = function(){
    console.log("this = ");    
    console.log(this);    // this指向person对象    
    console.log(this === person);
};

person.showName();
person.consThis();

输出如图:
图片描述

2.观察下person.showName属性引用的函数

// person.showName引用的是一个匿名函数,其name属性为""
console.log("person.showName.name=");
console.log(person.showName.name);
console.log(person.showName.name === "")
console.log("-----分割线-----");

// showName引用的匿名函数的prototype属性指向其原型对象
console.log("person.showName.prototype=");
console.log(person.showName.prototype);
console.log(person.showName.prototype === Function.prototype);        //不是Function构造函数的原型对象
console.log("-----分割线-----");

//其原型对象的constructor属性指向showName引用的匿名函数
console.log("person.showName.prototype.constructor=");        
console.log(person.showName.prototype.constructor);        
console.log("-----分割线-----");

// 此匿名函数的原型对象的__proto__属性指向Object构造函数的原型对象
console.log("person.showName.prototype.__proto__=");
console.log(person.showName.prototype.__proto__);
console.log(person.showName.prototype.__proto__ === Object.prototype);
console.log("-----分割线-----");

// showName引用的匿名函数的__proto__属性指向Function.prototype
console.log("person.showName.__proto__=");        //showName引用的是一个匿名函数,匿名函数本质上是由构造函数Function生成的
console.log(person.showName.__proto__);        //showName引用的是一个匿名函数,匿名函数本质上是由构造函数Function生成的
// 验证
console.log(person.showName.__proto__ === Function.prototype);    //true
console.log("-----分割线-----");

showName属性引用的匿名函数信息:
图片描述

输出如图:
图片描述

3.观察person对象涉及到的原型链

Object构造函数的原型链

// prototype属性指向Object构造函数的原型对象
console.log("Object.prototype=");
console.log(Object.prototype);
// Object构造函数的原型对象的constructor属性指向其本身
console.log("Object.prototype.constructor=");
console.log(Object.prototype.constructor);
console.log("-----分割线-----");

//所有构造函数(内置及自定义)的__proto__属性都指向Function构造函数的原型对象
// __proto__指向Function构造函数的原型对象,即Function.prototype
console.log("Object.__proto__=");
console.log(Object.__proto__);    
console.log(Object.__proto__ === Function.prototype);    
console.log("-----分割线-----");

图片描述

person对象的原型链

// person是对象,非函数,所以没有prototype属性
console.log("person.prototype=");
console.log(person.prototype);
console.log("-----分割线-----");

// 所有对象的__proto__都指向其构造器的prototype
// person的__proto__属性指向构造函数的原型对象,即Object构造函数的原型对象
console.log("person.__proto__=");
console.log(person.__proto__);
// 验证
console.log(person.__proto__ === Object.prototype)    //true
console.log("-----分割线-----");

图片描述

构造函数的原型对象的类型

// Function.prototype的类型为function
console.log("typeof Function.prototype:");
console.log(typeof Function.prototype);    
// 除了Function构造函数的原型对象为function,其他构造函数的原型对象类型为object
// Object.prototype的类型为object
console.log("typeof Object.prototype:");
console.log(typeof Object.prototype);
// Array.prototype的类型为object
console.log("typeof Array.prototype:");
console.log(typeof Array.prototype);
console.log("-----分割线-----");

图片描述

Function构造函数的原型对象和Object构造函数的原型对象的关系

// Function.prototype.__proto指向Objcet.prototype
console.log("Function.prototype.__proto__=");
console.log(Function.prototype.__proto__);
// 验证
console.log(Function.prototype.__proto__ === Object.prototype);    //true
// 这说明所有的构造器也都是一个普通JS对象,可以给构造器添加/删除属性等。同时它也继承了Object.prototype上的所有方法:toString、valueOf、hasOwnProperty等。
console.log("-----分割线-----");

// Object.prototype.__proto指向null
console.log("Object.prototype.__proto__=");
console.log(Object.prototype.__proto__);
// 原型链到顶了,为null。

原型分析

  • 所有构造器/函数的_ proto _都指向Function.prototype

  • 所有对象的_ proto _都指向其构造器的prototype

  • 除了Function构造函数的原型对象为function,其他构造函数的原型对象类型为object

  • 所有的构造器也都是一个普通JS对象,可以给构造器添加/删除属性等。同时它也继承了Object.prototype上的所有方法:toString、valueOf、hasOwnProperty等。

  • person对象是由Object构造函数创建而成的,所以它的_ proto _指向Object.prototype

原型链图

person对象的原型链图:
图片描述

person对象showName属性引用的匿名函数的原型链图:
图片描述

相关文章:

  • iOS边练边学--通知机制和键盘处理小练习
  • PHP CodeBase: 生成N个不重复的随机数
  • 解析stm32的时钟
  • BZOJ 1001 狼抓兔子 (网络流最小割/平面图的对偶图的最短路)
  • Material Design 控件
  • ARCproject中加入非ARC文件,或者非ARC环境中加入ARC文件
  • IOS开发UI篇--IOS动画(Core Animation)总结
  • css中的单位
  • 关于退运美国转基因玉米含有MRI 162转基因成分的质疑
  • Shiro基于组织机构的登录验证
  • maven部署构建到私服
  • Android 发送短信总结
  • 笔记 - 10.4、HTML - CSS滤镜笔记
  • Java BigDecimal详解
  • javap的使用
  • @jsonView过滤属性
  • [deviceone开发]-do_Webview的基本示例
  • [译] React v16.8: 含有Hooks的版本
  • 【剑指offer】让抽象问题具体化
  • 【译】React性能工程(下) -- 深入研究React性能调试
  • Apache的80端口被占用以及访问时报错403
  • CentOS7 安装JDK
  • ES6系列(二)变量的解构赋值
  • JavaScript函数式编程(一)
  • JS数组方法汇总
  • Laravel 中的一个后期静态绑定
  • Markdown 语法简单说明
  • PaddlePaddle-GitHub的正确打开姿势
  • PHP CLI应用的调试原理
  • puppeteer stop redirect 的正确姿势及 net::ERR_FAILED 的解决
  • WebSocket使用
  • 工作踩坑系列——https访问遇到“已阻止载入混合活动内容”
  • 前嗅ForeSpider中数据浏览界面介绍
  • 区块链技术特点之去中心化特性
  • 让你的分享飞起来——极光推出社会化分享组件
  • 深度学习入门:10门免费线上课程推荐
  • 使用Gradle第一次构建Java程序
  • 手写一个CommonJS打包工具(一)
  • 微信支付JSAPI,实测!终极方案
  • 运行时添加log4j2的appender
  • mysql面试题分组并合并列
  • 阿里云服务器如何修改远程端口?
  • 积累各种好的链接
  • ​Python 3 新特性:类型注解
  • (14)Hive调优——合并小文件
  • (DFS + 剪枝)【洛谷P1731】 [NOI1999] 生日蛋糕
  • (Matlab)基于蝙蝠算法实现电力系统经济调度
  • (多级缓存)多级缓存
  • (力扣)1314.矩阵区域和
  • (论文阅读32/100)Flowing convnets for human pose estimation in videos
  • (南京观海微电子)——COF介绍
  • (牛客腾讯思维编程题)编码编码分组打印下标(java 版本+ C版本)
  • (已解决)报错:Could not load the Qt platform plugin “xcb“
  • (转)负载均衡,回话保持,cookie
  • (转载)PyTorch代码规范最佳实践和样式指南