- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我读了一篇关于在类层次结构中保持数据私有(private)的文章 here .我这样做的方式不同。我将工厂函数与 Object.setPrototypeOf(obj, prototype) 一起使用.
为什么我的做法不被认为是好的做法?
这是我的做法:
我不需要公共(public)变量,所以我用工厂函数创建了我的狗对象:
const makeDog = (name) => {
return { bark: () => { console.log(name) } }
}
const myDog = makeDog("sniffles")
myDog.bark() // "sniffles"
所有动物都可以吃,我希望我的狗继承 Animal:
const makeAnimal = () => {
let numTimesEat = 0
return {
eat: (food) => {
numTimesEat += 1
console.log( "I eat " + food.toString() )
},
get numTimesEat() { return numTimesEat }
}
}
const myAnimal = makeAnimal()
myDog 将委托(delegate)给 myAnimal 吃东西:
Object.setPrototypeOf(myDog, myAnimal)
现在我可以:
myDog.eat("shoe") // "I eat shoe"
console.log( myDog.numTimesEat ) // 1
myDog.bark() // "sniffles"
请注意,myDog.numTimesEat
应该是指 myDog 进食的次数。
附注我知道你可以通过类(class)来做到这一点:
class Animal {
constructor() {
this.numTimesEat = 0;
}
eat(food) {
this.numTimesEat += 1;
console.log( "I eat " + food.toString() );
}
}
class Dog extends Animal {
constructor(myName) {
super();
this.name = myName;
}
bark() {
console.log( this.name );
}
}
const dog2 = new Dog("sniffles");
dog2.eat("shoe"); // "I eat shoe"
console.log( dog2.numTimesEat ); // 1
console.log( dog2.name ); // "sniffles"
dog2.bark(); // "sniffles"
但是 class 关键字似乎最终会在我的对象上生成公共(public)变量。如果我尝试使用类似 these 的技术它看起来有点丑(我想下划线语法看起来还不错,但它并不是真正的私有(private))。
解决方案:
如果我们创建 10 只狗,它们都使用相同的 Animal 作为其原型(prototype),则共享“let numTimesEat”。如果一只狗吃一次,您不希望 numTimesEat 为 10。
因此除了设置原型(prototype) 10 次(重复执行的缓慢操作)之外,您还必须为这 10 只狗创建 10 只动物以委托(delegate)给它们。
更新:相反,您可以将所有内容放在新创建的对象上
const Dog = function(name) {
let that = Animal()
that.bark = () => { console.log(name) }
return that
}
const Animal = function() {
let numTimesEat = 0
return {
eat: (food) => {
numTimesEat += 1
console.log( "I eat " + food.toString() )
},
get numTimesEat() { return numTimesEat }
}
}
const lab = new Dog("sniffles")
lab.bark() // sniffles
lab.eat("food") // I eat food
lab.numTimesEat // 1
这比 trying to do OOP in Javascript 干净多了.
最佳答案
简单。
如果我们在 Javascript 中没有原型(prototype)或this
,我们仍然可以愉快地进行 OOP 和继承,就像您描述的那样:
甚至没有隐藏状态:
现在有人可能会说'嘿,我的对象有 2 个状态变量和 10 个方法,每当我想要一个新的对象实例时都需要复制 10 个方法,这不是很浪费吗?'
然后我们可以说“是的,让我们实际上在同一类型的所有对象之间共享函数”。但这会带来两个问题:
this
作为隐藏参数传递给每个方法)...因此我们到达了标准 JS OOP 的当前状态。在此过程中,我们牺牲了在方法闭包中保持私有(private)状态的可能性(因为每个对象只有一种方法)。
因此,您实际上不想去那里,而是停留在每个对象都有自己的方法副本的第一个位置。很好,您可以拥有私有(private)状态!但是为什么你此时甚至需要原型(prototype)呢?他们不能达到最初的目的。一个对象的 bark()
与另一个对象的 bark()
完全无关,它们有不同的闭包,它们不能共享。对于任何继承的方法都是一样的。
在这种情况下,因为您已经复制了所有方法,所以您也可以将它们保留在对象本身中。向每个对象添加(不同的)原型(prototype)并在其中放置父类(super class)方法不会给您带来任何好处,它只是为方法调用增加了一层间接性。
关于javascript - 为什么在构建 Javascript 类层次结构时 Object.setPrototypeOf 不鼓励/效率低下?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45120928/
MDN 暗示使用 .setPrototypeOf() 会对代码的 future 性能产生不良影响。 我还阅读了一些关于为什么更改对象的 [[Prototype]] 会降低性能的问题。但是没有一个答案真
因此,我一直在快速了解 JavaScript 的一些新功能,并且一直在阅读有关 Object.setPrototypeOf() 的内容。我从 MDN 中遇到了这段代码它处理从常规对象继承。但是我对他们
所以我读了一些关于 subclassing Array in JavaScript 的博客文章、SO 线程和其他讲座。关于该主题的普遍观点是,没有办法创建具有某种缺点的子类。 在尝试一些事情时,我自己
我正在阅读express 4的源代码,它是相当新的,其中有以下代码: var proto = module.exports = function(options) { var opts = opt
我读了here我们可以使用Object.create来实现继承 这是一个继承自 Shape 的 Rectangle 示例 function Shape() {} function Rectangle(
我想实现 polyfill Object.setPrototypeOf,如下所述: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Re
我有以下代码确保 Promise.reject 返回一个实际的错误对象: export const rejectWithAnError = function(error) { if(error.c
我一直在研究一种类似 ORM 的方法来在 Node.JS 应用程序中处理我的模型,我对它的工作方式很满意;但是,我想知道是否有更好的方法来处理重置构造函数原型(prototype)的问题。 我想要的行
显然使用 __proto__ 属性仍然是操作原型(prototype)链的主要方式,尽管这不符合标准并且 IE 不支持它。虽然您也可以通过使用 new 构造函数构造继承,但与 __proto__ 属性
我只是想知道替换 __proto__ 的主要原因是什么功能 getPrototypeOf/setPrototypeOf ? 提前致谢 最佳答案 MDN explains it pretty well.
当我阅读 MDN( https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes#Sub_classing_wit
我知道两种继承函数构造函数的方法。 选项 1 Object.create function x(x, y) { this.x = x; this.y = y; } x.prototype.XD
MDN 有一个关于修改代码原型(prototype)的巨大警告: Changing the [[Prototype]] of an object is, by the nature of how mo
我正在尝试使用 Object.setPrototypeOf 来重新设置已转换为 JSON 并返回为对象的对象的原型(prototype)。但是,instanceof 似乎没有按我预期的方式工作: cl
让我们考虑以下代码, let f = function () { this.a = 1; this.b = 2; } let o = new f(); f.prototype.c = 3;
我读了一篇关于在类层次结构中保持数据私有(private)的文章 here .我这样做的方式不同。我将工厂函数与 Object.setPrototypeOf(obj, prototype) 一起使用.
我不确定标题是否真的有意义,但我正在尝试将函数原型(prototype)设置为“子类”原型(prototype)。 举个例子; 我尝试做的是:我有一个 user 和 paidUser 。 paidUs
我有一个 Node.js 项目,其中有我自己的自定义错误: 'use strict'; require('util').inherits(module.exports, Error); functio
我有一个 Angular 应用程序,它运行良好。但是,在使用 yarn upgrade 升级依赖项后,我收到了这个神秘错误: main.bundle.js:6779 Uncaught TypeErro
假设我有一个具有speak 功能的动物对象: function speak() { console.log(this.sound) } let animal = { speak } 我有一只狗
我是一名优秀的程序员,十分优秀!