翼度科技»论坛 编程开发 JavaScript 查看内容

创建型-原型模式

3

主题

3

帖子

9

积分

新手上路

Rank: 1

积分
9
定义

 使用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。原型模式是一种对象创建型模式---百科。
 通俗的说就是原型模式是一种创建型设计模式,指定某个对象(通过某种方式)得到一个新的对象,在内存中拥有新的地址,得到的对象与原对象是是相互独立的,即得到跟原对象一样的对象
 当我们需要两个一模一样的实例时,使用原型模式非常方便,如果不使用原型模式,按照构造函数的方式初始化对象,我们需要传两次一模一样的参数如:
  1. const dog = new BydCard('byd', '汉', '30w', '2023款')
  2. const dog_copy = new BydCard('byd', '汉', '30w', '2023款')
  3. // 使用原型模式
  4. const dog_copy1 = Object.create(dog)
复制代码
实现思路

  通过目标对象得到一个全新的新对象,使新对象也具备跟目标对象一样的能力,这种一般思路有两种

  • 深拷贝
  • 指针引用:自身对象找不到,通过内部属性引用到目标对象上去找类似链表结构的next 指针
其中大多数后台语言如java 有相关克隆接口规范,javaScript 是通过第二种方式来实现的。

javaScript 中的原型模式

  在原型模式下,当我们想要创建一个对象时,会先找到一个对象作为原型,然后通过克隆原型的方式来创建出一个与原型一样(共享一套数据/方法)的对象。在 JavaScript 里,Object.create方法就是原型模式的天然实现——准确地说,只要我们还在借助Prototype来实现对象的创建和原型的继承,那么我们就是在应用原型模式。
  有的设计模式资料中会强调,原型模式就是拷贝出一个新对象,认为在 JavaScript 类里实现了深拷贝方法才算是应用了原型模式。事实上在 JavaScript 中,通过指针的方式也可以得到目标对象、属性、方法的共享。克隆(深度拷贝)是实现这个目的的方法,但不是唯一的方法,也不是javaScript 的目的。
通过指针来引用,然后动态执行的时候绑定上下文 this,这样就不会造成实例之间的错乱,我觉得这也是this 被设计成动态绑定的原因之一。

原型模式-编程范式

  原型模式不仅是一种设计模式,它还是一种编程范式(programming paradigm),是 JavaScript 面向对象系统实现的根基,原型编程范式的体现就是基于原型链的继承。即便现在es6+ 推出了class 关键字,支持了类的写法。引入的 JavaScript 类本质上还是基于原型的继承的语法糖(class 只是一个语法糖)。类语法不会为 JavaScript 引入新的面向对象的继承模型。 当我们尝试用 class 去定义一个 Dog 类时:
  1. class Dog {
  2.   constructor(name ,age) {
  3.    this.name = name
  4.    this.age = age
  5.   }
  6.   
  7.   eat() {
  8.     console.log('肉骨头真好吃')
  9.   }
  10. }
复制代码
其实完全等价于写了这么一个构造函数:
  1. function Dog(name, age) {
  2.   this.name = name
  3.   this.age = age
  4. }
  5. Dog.prototype.eat = function() {
  6.   console.log('肉骨头真好吃')
  7. }
复制代码
原型链核心点

  每个构造函数都拥有一个prototype属性,它指向构造函数的原型对象,这个原型对象中有一个 constructor 属性指回构造函数;每个实例都有一个内部属性__proto__属性,当我们使用构造函数去创建实例时,实例的__proto__属性就会指向构造函数的原型对象。
  1. // 输出"肉骨头真好吃"
  2. dog.eat()
  3. // 输出"[object Object]"
  4. dog.toString()
复制代码
明明没有在 dog 实例里手动定义 eat 方法和 toString 方法,它们还是被成功地调用了。这是因为当我试图访问一个 JavaScript 实例的属性、方法时,它首先搜索这个实例本身;当发现实例没有定义对应的属性、方法时,它会转而去搜索实例的原型对象;如果原型对象中也搜索不到,它就去搜索原型对象的原型对象,这个搜索的链表就叫做原型链。


Object 是所有的基类,其中Object.prototype指向null,这样原型链就有终点了,而不是无脑的一直下去。
原型链其他关键点:

  • 所有函数(普通函数,构造函数,内置的函数)都是内置函数(类)Function 的实例,所以存在函数.__proto__ === Function.prototype 所有函数都可以直接调用Function原型上的方法(call / apply /bind)
  • Function 确实很厉害,他不仅是函数的类,还是自己的类。函数是Function 的实例,Function 也是Function 的实例 Object.__proto__ === Function.prototypeFunction.__proto__===Function.prototype
  • 对象的原型链最终指向Object.prototype, object.prototype._proto_ 指向null
如下代码验证了这些结论:
  1. function sayHi () {
  2.     // console.log('hello joel')
  3. }
  4. // 所有函数都是Function 的实例即函数也是对象,
  5. // 所以存在函数.__proto__ === Function.prototype
  6. console.log(sayHi.__proto__ === Function.prototype) // true
  7. console.log(Object.__proto__ === Function.prototype) // true
  8. console.log(String.__proto__ === Function.prototype) // true
  9. console.log(Array.__proto__ === Function.prototype) // true
  10. console.log(Number.__proto__ === Function.prototype) // true
  11. console.log(Symbol.__proto__ === Function.prototype) // true
  12. // Function.prototype 内部属性又指向Object的原型对象
  13. console.log(Function.prototype.__proto__ === Object.prototype) // true
  14. //  Function 也是Function 的实例
  15. console.log(Function.__proto__ === Function.prototype)
  16. // 对象最终指向object的原型
  17. console.log(new sayHi().__proto__ instanceof Object) // true
  18. console.log(new sayHi().__proto__ === sayHi.prototype) // true
  19. console.log(Array.prototype.__proto__ === Object.prototype) // true
  20. console.log(Object.__proto__.__proto__ === Object.prototype) // true
  21. // 内置的array,string,number,object 等都是构造函数,同时也是对象
  22. console.log(typeof Array) // function
  23. console.log(typeof Object) // function
  24. // 通过原型链找到object.prototype 上的方法
  25. sayHi.valueOf()
复制代码
小结


  • 原型是 JavaScript 面向对象系统实现的根基,在这里更像是一种编程范式
  • 在JavaScript 中原型模式无处不在,只要使用原型的模型创建对象就是在使用原型模式
  1. Object.__proto__ === Function.prototype
  2. Function.__proto__=== Function.prototype
  3. Function.prototype.__proto__ === Object.prototype
  4. Object.prototype.__prto__ === null
复制代码
 

来源:https://www.cnblogs.com/longbensong/p/17198551.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具