JavaScript ES5 中原型的概念

news/2024/10/20 1:38:27

对象和函数的原型

认识对象的原型

◼ JavaScript当中每个对象都有一个特殊的内置属性 [[prototype]],这个特殊的对象可以指向另外一个对象。
◼ 那么这个对象有什么用呢?当我们通过引用对象的属性key来获取一个value时,它会触发[[Get]]的操作;这个操作会首先检查该对象是否有对应的属性,如果有的话就使用它;如果对象中没有改属性,那么会访问对象[[prototype]]内置属性指向的对象上的属性;◼ 那么如果通过字面量直接创建一个对象,这个对象也会有这样的属性吗?如果有,应该如何获取这个属性呢?答案是有的,只要是对象都会有这样的一个内置属性;
◼ 获取的方式有两种:方式一:通过对象的__proto__ 属性可以获取到(但是这个是早期浏览器自己添加的,存在一定的兼容性问题);方式二:通过Object.getPrototypeOf 方法可以获取到;
案例:var obj = {name : 'hdc',age : 18
}
console.log(obj)
// 获取原型的方式
// 浏览器自己加的属性
console.log(obj.__proto__)
//标准方法
console.log(Object.getPrototypeOf(obj))console.log(obj.__proto__ === Object.getPrototypeOf(obj))//true

函数的原型prototype

◼ 那么我们知道上面的东西对于我们的构造函数创建对象来说有什么用呢?它的意义是非常重大的,接下来我们继续来探讨;
◼ 这里我们又要引入一个新的概念:所有的函数都有一个prototype的属性(注意:不是__proto__)
◼ 你可能会问题,老师是不是因为函数是一个对象,所以它有prototype的属性呢?不是的,因为它是一个函数,才有了这个特殊的属性;而不是它是一个对象,所以有这个特殊的属性
案例:<script>var obj = {}function foo(){      }//1.将函数看成普通的obj对象时是具备__proto__(隐式原型)//作用:查找key对的value时最终会找到原型身上console.log(obj.__proto__)console.log(foo.__proto__)//2.将函数看成是一个函数是,它具备prototype(显示原型)//所有函数都有一个属性prototype//作用:用来构建对象时给对象设置隐式原型的console.log(foo.prototype)// console.log(obj.prototype)//对象没用这个属性</script>

new、constructor

再看new操作符

◼ 我们前面讲过new关键字的步骤如下:1.在内存中创建一个新的对象(空对象);2.这个对象内部的[[prototype]]属性会被赋值为该构造函数的prototype属性;
◼ 那么也就意味着我们通过Person构造函数创建出来的所有对象的[[prototype]]属性都指向Person.prototype:1.创建空对象var obj = {}
2.将这个空对象赋值给thisthis = obj
3.将函数的显示原型赋值给这个对象作为他的隐式原型
obj.__proto__ = Preson.prototype
4.执行函数中的代码
5.如果没有返回非空对象将这个对象默认返回案例:<script>function Foo(){//1.创建空的对象// 2. 将Foo的prototype原型(显示原型)赋值给空的对象的__proto__(隐式原型)}console.log(Foo.prototype)var f1 = new Foo()var f2 = new Foo()var f3 = new Foo()var f4 = new Foo()var f5 = new Foo()console.log(f1.__proto__)console.log(Foo.prototype === f1.__proto__)//trueconsole.log(f3.__proto__ === f5.__proto__)//true</script>

constructor属性

◼ 事实上原型对象上面是有一个属性的:constructor默认情况下原型上都会添加一个属性叫做constructor,这个constructor指向当前的函数对象;
案例:// 非常重要的属性:constructor 指向的是函数Personvar PersonProtoType = Person.prototypeconsole.log(PersonProtoType)console.log(PersonProtoType.constructor)console.log(PersonProtoType.constructor === Person)  //trueconsole.log(Person.name)//Personconsole.log(PersonProtoType.constructor.name)  //Person

将所有的函数定义放到了显式原型上

    <script>function Student(name,age,sno){this.name = namethis.age = agethis.sno = sno// 方式一 : 编写函数会创建多个函数对象// this.running = function(){//   console.log(this.name + "在跑步")// }// this.eating = function(){//   console.log(this.name + "在吃饭")// }// this.studying = function(){//   console.log(this.name + "在学习")// }}//将函数放在原型上// 当我们多个对象之间拥有共同的值时我们可以将它放在构造函数对象的显式原型// 由构造函数创建出来的所有对象,都会共享这些属性Student.prototype.running = function(){console.log(this.name + "在跑步")}Student.prototype.eating = function(){console.log(this.name + "在吃饭")}var stu1 = new Student("hdc",21,111)var stu2 = new Student("kobe",30,112)var stu3 = new Student("james",25,113)//默认时 创建多个相同的函数对象console.log(stu1.running == stu3.running)//false// 隐式原型的作用// 1> stu1的隐式原型是Student.prototype对象// 2> stu1.running 查找//      * 先在自己的身上找//      * 去原型里面查找stu1.running()stu3.running()stu3.eating()</script>

重写原型对象

◼ 如果我们需要在原型上添加过多的属性,通常我们会重写整个原型对象:
◼ 前面我们说过, 每创建一个函数, 就会同时创建它的prototype对象, 这个对象也会自动获取constructor属性;而我们这里相当于给prototype重新赋值了一个对象, 那么这个新对象的constructor属性, 会指向Object构造函数, 而不是Person构造函数了
案例:<script>function Person(){}console.log(Person.prototype)//在原有的原型对象上添加新的属性Person.prototype.message = "Hello Person"Person.prototype.running = function(){}// 直接赋值一个新的原型对象  缺点没有constructor对象Person.prototype = {message : "Hello Person",info:{name:"hdc",age:30},running:function(){},eatting:function(){},// 手动添加constructor对象constructor:Person}// 新建实例对象var p1 = new Person()console.log(p1.message)</script>

原型对象的constructor

  ◼ 如果希望constructor指向Person,那么可以手动添加:◼ 上面的方式虽然可以, 但是也会造成constructor的[[Enumerable]]特性被设置了true.默认情况下, 原生的constructor属性是不可枚举的.如果希望解决这个问题, 就可以使用我们前面介绍的Object.defineProperty()函数了.案例:console.log(Person.prototype) // {message: 'Hello Person', info: {…}, running: ƒ, eatting: ƒ}console.log(Object.keys(Person.prototype))  //(5) ['message', 'info', 'running', 'eatting', 'constructor']// 默认情况下, 原生的constructor属性是不可枚举的.// 如果希望解决这个问题, 就可以使用我们前面介绍的Object.defineProperty()函数了.Object.defineProperty(Person.prototype,"constructor",{enumerable:false,value:Person})console.log(Object.keys(Person.prototype))  //(4) ['message', 'info', 'running', 'eatting']

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.ryyt.cn/news/73643.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈,一经查实,立即删除!

相关文章

Codeforces Round 979 (Div. 2)题解记录

比赛链接:https://codeforces.com/contest/2030 A. A Gift From Orangutan 肯定最小值和最大值放前面最好,答案得解#include<iostream>#include<string.h>#include<map>#include<vector>#include<set>#include<unordered_set>#include&l…

AI工人操作行为流程规范识别系统

AI工人操作行为流程规范识别系统利用高清监控摄像头覆盖现场作业区域,AI工人操作行为流程规范识别系统通过图像识别和深度学习技术对作业人员的操作行为进行实时分析。AI工人操作行为流程规范识别系统能够准确识别工人的操作行为是否符合作业标准规定的流程和合规SOP,并根据设…

高级程序语言课第三次作业

2024高级语言程序设计:https://edu.cnblogs.com/campus/fzu/2024C 高级语言程序设计课程第三次个人作业:https://edu.cnblogs.com/campus/fzu/2024C/homework/13284 学号:102400115 姓名:洪育豪 作业内容:编写并运行书本第4章4.8编程练习题目中的第2题到第4题,第6题到第8题 编…

传送带下料口堵塞识别检测系统

传送带下料口堵塞识别检测系统利用AI视觉识别算法,传送带下料口堵塞识别检测系统通过现场监控摄像头对传送带的运输物料过程进行实时分析和识别。传送带下料口堵塞识别检测系统能够准确判断下料口是否出现堵塞现象,并及时抓拍有关图像进行记录。传送带下料口堵塞识别检测系统…

React/Vue 实现的前端应用, java/Go/Python 实现的后端应用,前后端分离的应用部署的最佳实践

前后端分离的应用(React 前端 + Java 后端)在部署过程中,需要考虑性能、扩展性、安全性、以及维护方便性等多个方面。下面我将详细介绍前后端分离应用的最佳实践,从架构设计、构建和打包、部署策略、CI/CD 集成、安全性措施等几个角度来描述。 微服务架构图示例壹.总体概述…

gradle配置代理

下载gradle项目 访问:https://start.spring.io/如上图所示,生成代码 配置代理服务器 买个国外的节点,使用 xshell 带代理方式连接,会暴露出 socks://localhost:1080建议开启 BBR 拥塞控制 # 要确保 linux 内核版本是4.9或更高,否则后面不用做了 uname -r # 加载 TCP BBR 模…

《使用Gin框架构建分布式应用》阅读笔记:p88-p100

《用Gin框架构建分布式应用》学习第6天,p88-p100总结,总计13页。 一、技术总结 1.MongoDB CRUD操作 (1)InsertOne(), InsertMany() (2)Find() (3)UpdateOne, UpdateMany() (4)DeleteOne(), DeleteMany() 2.MongoDB primitive p96,recipe.ID = primitive.NewObjectID() 中的…

在blender中打开pmx文件

适用blender版本: 3.6 - 4.0 - 4.1 - 4.2 等 本人使用的blender版本为3.6 和 4.2 这里用3.6作案例下载cats插件在github中查找cats-blender-plugin 比如说这个:https://github.com/absolute-quantum/cats-blender-plugin下载最新的插件 注意: 插件版本只对应相应的blender版本…