看了《JavaScript 高级程序设计》,有个关于动态原型模式的问题

2016-07-11 10:07:37 +08:00
 cheroky

书上介绍了动态原型模式的代码如下:

function Person(name,age,job) {
    //property
    this.name = name;
    this.age = age;
    this.job = job;
    //method
    if(typeof this.sayName != "function") {
        Person.prototype.sayName = function() {
            console.log(this.name);
        }
        
        //Person.prototype = {
        //    sayName: function() {
        //        console.log(this.name);
        //    }
        //} //这样的写函数的方式报错说没有 sayName 这个函数
    }
}

当我用注释的 sayName 写法时会提示找不到 sayName 这个函数,为什么??
两种方式有什么异同吗?我记得书上提过第二种只是对第一种声明变量多的简便写法。
3233 次点击
所在节点    JavaScript
10 条回复
judasnow
2016-07-11 10:13:33 +08:00
你直接把 prototype 给覆盖了。放到构造函数外面就行了。
Parabolazz
2016-07-11 10:27:18 +08:00
你看高程 156 页,用对象字面量的方法相当于重写了整个原型。切断了构造函数和最初的原型之间的联系。
我的理解是你这时 new 一个新实例,实例的原型并没有 sayName 方法,因为 sayName 方法保存在新的原型对象上了。
shyling
2016-07-11 10:31:17 +08:00
把注释的部分放到 Person 外面
coolzjy
2016-07-11 10:33:29 +08:00
judasnow
2016-07-11 11:05:08 +08:00
@judasnow 研究了下,忽略我的第一句话吧 233
songz
2016-07-11 11:25:13 +08:00
Person.prototype.sayName
learnshare
2016-07-11 11:32:57 +08:00
不要覆盖 prototype ,用 #6 的方式,扩展 prototype 。
palmers
2016-07-11 15:07:41 +08:00
据我的分析是这样的: 你注释的内容改变了对象的原型对象, 所以,通过 Person `new`操作符实例化的对象的原型是 Object 不再是 Person.prototype 了,然而, sayName 函数并不在 Person 对象的原型上,所以找不到,注释代码中 sayName 函数实际上是在通过 new Person()得到的对象的构造函数的原型上,也就是 Person.prototype.constructor.__proto__ ;
Cytrs
2016-07-12 08:18:05 +08:00
之前刚好看到了一篇文章 https://segmentfault.com/a/1190000000602050
Sparetire
2016-07-12 14:13:52 +08:00
准确说应该是创建的第一个对象没有 sayName 方法,之后创建的对象都有 sayName 方法,可以试试
因为创建第一个对象的时候的原型对象没有 sayName 方法,之后每次创建对象都重写了原型,且这些原型都有 sayName 方法,所以只有第一个对象没有 sayName ,之后的都有

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/291627

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX