构造函数的Prototype上定义的方法确实可以通过对象直接调用,而且代码是共享的。我表示我不懂。太难理解了,艹。
在Javascript中,prototype不但能让对象共享自己的财富,而且prototype还有寻根问祖的天性。
当从一个对象那里读取属性或调用方法时,如果该对象自身不存在这样的属性或方法,就会去自己关联的prototype对象里寻找。如果prototype没有,
又会去prototype自己关联的前辈prototype那里寻找,直到找到或追溯过程结束为止。

我们先从Javascript中的构造函数说起,

<script type="text/javascript">
function Person(name){//带参数的构造函数
this.name = name;//定义并初始化name属性
this.SayHello = function(){//定义对象方法SayHello
alert("Hello,I'm " + this.name);
}
} function Employee(name,salary)//子构造函数
{
Person.call(this,name);//调用父构造函数
this.salary = salary;//添加并初始化salary属性
this.ShowMeTheMoney = function(){
//添加对象方法ShowMeTheMoney
alert(this.name+" $"+this.salary);
}
}
var BillGates = new Person("Bill Gates");//创建Person对象
var SteveJobs = new Employee("Steve Jobs",9999);//创建Employee类的SteveJobs对象
BillGates.SayHello();//输出I'm Bill Gates
SteveJobs.SayHello();//输出I'm Steve Jobs
SteveJobs.ShowMeTheMoney();//Steve Jobs $9999 alert(BillGates.constructor == Person);//true
alert(SteveJobs.constructor == Employee);//true
alert(BillGates.SayHello == SteveJobs.SayHello);//false
</script>

这里的Employee继承了Person,包括属性设置和方法。但是这里的SayHello方法却不是一个方法,而是在Employee中新开辟的一个方法。相当于复制过来的方法。

而是用prototype就不是这样了。prototype可以让Employee共享Person的原型方法。

如果Person创建两个对象,那么它们调用同一个方法吗?

我们来看代码:

<script type="text/javascript">
function SayHello(){
alert("Hello,I'm "+this.name);
}
function Person(name){
this.name = name;
this.SayHello = SayHello;
}
var BillGates = new Person("Bill Gates");
var SteveJobs = new Person("Steve Jobs");
BillGates.SayHello();//Hello,I'm Bill Gates
SteveJobs.SayHello();//Hello,I'm Steve Jobs
alert(BillGates.SayHello==SteveJobs.SayHello);//true
</script>

结果发现,是的,它们的确调用一个方法。而不是像Employee那么复制出的新方法。

上述代码转化成原型模式,也很简单,如下所示,先设置属性,再通过原型定义方法:

<script type="text/javascript">
function Person(name){
this.name = name;//设置对象属性,每个对象各有一份属性数据
}
Person.prototype.SayHello = function(){//构造函数的Prototype上定义的方法确实可以通过对象直接调用,而且代码是共享的。我表示我不懂。太难理解了,艹。
alert("Hello,I'm "+this.name);
}
var BillGates = new Person("Bill Gates");
var SteveJobs = new Person("Steve Jobs");
BillGates.SayHello();//Hello,I'm Bill Gates
SteveJobs.SayHello();//Hello,I'm Steve Jobs
alert(BillGates.SayHello==SteveJobs.SayHello);//true </script>

"原型继承"是慈祥而又严厉的。原型对象将自己的属性和方法无私地贡献给孩子们使用,也并不强迫孩子们必须遵从,允许一些顽皮孩子按自己的兴趣和爱好独立行事。
我们来进一步研究原型prototype的特点,

<script type="text/javascript">
function Person(name){
this.name = name;
}
Person.prototype.company = "Microsoft";//原型的属性
Person.prototype.SayHello = function(){//原型的方法
alert("Hello,I'm "+this.name+" of "+this.company);
}
var BillGates = new Person("Bill Gates");
BillGates.SayHello();//由于继承了原型的东西,规规矩矩输出:Hello,I'm Bill Gates
var SteveJobs = new Person("Steve Jobs");
SteveJobs.company = "Apple";//设置了自己的company属性,掩盖了原型的company属性
SteveJobs.SayHello = function(){//实现了自己的SayHello方法,掩盖了原型的SayHello方法
alert("Hi, "+this.name+" like "+this.company+",ha ha");
}
BillGates.SayHello();//没有被改变,还是输出Hello,I'm Bill Gates of Microsoft
SteveJobs.SayHello();//原有的都被覆盖掉了,输出Hi, SteveJobs like Aplle,ha ha
</script>

这里原型中的属性和方法可以被覆盖掉。但原型自身的并没有受到影响。

对象可以掩盖原型对象的那些属性和方法,一个构造函数原型对象也可以掩盖上层构造函数原型对象既有的属性和方法。
我们可以随时给原型对象动态添加新的属性和方法,从而动态扩展基类的功能特性。我们来看下面的代码:

<script type="text/javascript">
function Person(name){//构造函数
this.name = name;
}
Person.prototype.SayHello = function(){//原型方法
alert("Hello,I'm "+this.name);
}
var BillGates = new Person("Bill Gates");//建立对象
BillGates.SayHello();
Person.prototype.Retire = function(){//对象后,动态扩展原型的方法
alert("Poor "+this.name+" ,byebye~!");
}
BillGates.Retire();//扩展的方法可以立刻使用
</script>

这里后续动态添加的方法,之前的对象已然可以使用,就是这么神奇。

最后我们用一个稍微复杂的例子来说明原型的使用:

<script type="text/javascript">
function Person(name){//构造函数
this.name = name;
}
Person.prototype.SayHello = function(){//原型方法
alert("Hello,I'm "+this.name);
}
//子类构造函数
function Employee(name,salary){
//Person.call(this,name);//调用上级的构造函数
this.name = name;
this.salary = salary;//扩展的成员属性
} //子类构造函数需要用上层构造函数来建立prototype对象
Employee.prototype = new Person();//继承了父类的prototype //子类定义自己的构造函数
Employee.prototype.ShowMeTheMoney = function(){
alert(this.name+" $"+this.salary);
} var BillGates = new Person("Bill Gates");
BillGates.SayHello();
//BillGates.ShowMeTheMoney();//不能用,因为Person的prototype里并没有这个方法 var SteveJobs = new Employee("Steve Jobs",1234);
SteveJobs.SayHello();//继承了Person的prototype方法
SteveJobs.ShowMeTheMoney();
alert(BillGates.SayHello == SteveJobs.SayHello);//True
</script>

这里的两个SayHello是同一个方法哦,也就是调用同一个方法产生的哦,跟之前的Employee复制过来的有区别哦。这点,读着可以细细体会其中的奥秘。

总之,原型prototype可以共享方法,"原型继承"是慈祥而又严厉的。原型对象将自己的属性和方法无私地贡献给孩子们使用,也并不强迫孩子们必须遵从,允许一些顽皮孩子按自己的兴趣和爱好独立行事。

悟透Javascript之 原型prototype的更多相关文章

  1. Javascript代码执行过程-《悟透Javascript》笔记

    本文摘录自李战老师<悟透Javascript>一书的部分章节,为适应博客发表作了一点点修改. 1) 预编译分析. JavaScript执行引擎将所有定义式函数直接创建为作用域上的函数变量, ...

  2. 悟透JavaScript

    要理解JavaScript,你得首先放下对象和类的概念,回到数据和代码的本原.前面说过,编程世界只有数据和代码两种基本元素,而这两种元素又有着纠缠不清的关系.JavaScript就是把数据和代码都简化 ...

  3. 悟透JavaScript(理解JS面向对象的好文章)

    引子 编程世界里只存在两种基本元素,一个是数据,一个是代码.编程世界就是在数据和代码千丝万缕的纠缠中呈现出无限的生机和活力. 数据天生就是文静的,总想保持自己固有的本色:而代码却天生活泼,总想改变这个 ...

  4. (转载)悟透JavaScript

    引子 编程世界里只存在两种基本元素,一个是数据,一个是代码.编程世界就是在数据和代码千丝万缕的纠缠中呈现出无限的生机和活力. 数据天生就是文静的,总想保持自己固有的本色:而代码却天生活泼,总想改变这个 ...

  5. javascript进阶-原型prototype

    一.javascript原型认识 很多编程语言都有类的概念,我们可以拿原型和类进行比较,看看它们之间的区别以及相同点在哪里. 1.类:类是一个具体事物的抽象所以类是一个抽象的东西,在面向对象中类可以用 ...

  6. 悟透JavaScript(二)

    初看原型 prototype源自法语,软件界的标准翻译为“原型”,代表事物的初始形态,也含有模型和样板的意义.JavaScript中的prototype概念恰如其分地反映了这个词的内含,我们不能将其理 ...

  7. 简单理解javascript的原型prototype

    原型和闭包是Js语言的难点,此文主要讲原型. 每一个方法都有一个属性是 prototype 每一个对象都有一个属性是 _proto_ 一旦定义了原型属性或原型方法,则所有通过该构造函数实例化出来的所有 ...

  8. [js高手之路]一步步图解javascript的原型(prototype)对象,原型链

    我们接着上文继续,我们通过原型方式,解决了多个实例的方法共享问题,接下来,我们就来搞清楚原型(prototype),原型链的来龙去脉. function CreateObj(uName) { this ...

  9. 悟透JavaScript (一)

    首先说明,这是别人写的一篇文章,写得很好,对理解JavaScript很有好处,所以转帖过来. 引子    编程世界里只存在两种基本元素,一个是数据,一个是代码.编程世界就是在数据和代码千丝万缕的纠缠中 ...

随机推荐

  1. angularjs 根据变量改变 动态加载模板

    directive return { restrict: 'E', replace: true, templateUrl: 'app/view/order.html', link: function ...

  2. window和Linux下的软链接

    window下开启软链接命令: mklink /J  "D:\IdeaProjects\bms_work\smartcity_govnet\base_web\target\base_web\ ...

  3. modelsim命令行仿真提示“vsim 不是内部或外部命令,也不是可运行的程序或批处理文件”的解决办法

    安装完modelsim后,用过命令行模式仿真,如“vsim -c -do run.do”,开始时是可以的. 后来偶然再用该仿真方式,发现命令行提示“vsim 不是内部或外部命令,也不是可运行的程序或批 ...

  4. JDK自带方法实现RSA非对称加密

    package jdbc.pro.lin; import java.security.InvalidKeyException; import java.security.Key; import jav ...

  5. portal开发"下拉框"“日期框”查询要怎么配置

    下面的这些是我今天的成果! 总的来说是一步一步摸索出来的!还是等感谢超哥的耐心指导,犯了一些错误! 1.比如在wd配置文件中中写id=“check_it_two”,在java中写成 checki_it ...

  6. 1. 连接字符串的创建 - Lazy.Framework从零开始设计自己的ORM架构

    开发初衷 注册了博客园已经有几个月了,却从来都没有上来过,本人大概从2010年开始就开始做.NET 方向的开发. 这个是我在博客园发布的第一个帖子. 主要就是说说最近在写的一个ORM架构. 本人接触的 ...

  7. [GDI+] 生成缩略图的类文件SmallImage (转载)

    直接看代码吧,大家可以直接复制使用 /// <summary> /// 类说明:SmallImage类, /// 编码日期:2012-08-20 /// 编 码 人: 苏飞 /// 联系方 ...

  8. 无法绑定到新的显示成员,参数名:newDisplayMember

    此问题不是网上说的 DisplayMember 等先后顺序问题,即使更换绑定数序,只是把错误覆盖而已(绑定的是对象的类名) ValueMember = "Id"; DisplayM ...

  9. shell脚本学习之$0,$?,$!等的特殊用法

    变量说明: $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process的PID $? 最后运行的命令的结束代码(返回值) $- 使用Set命令设定的Flag一览 ...

  10. IOS pop使用代理传值

    假如oneViewController页面push到OtherViewController页面,然后你想从OtherViewController页面pop到oneViewController页面的时候 ...