Javascript中的对象和原型(二)中我们提到,用构造函数创建的对象里面,每个对象之间都是独立的,这样就会降低系统资源的利用率,解决这样问题,我们就要用到下面提到的原型对象。

  一 原型对象

  原型对象实际上就是构造函数的一个实例对象,和普通的实例对象没有本质上的区别。可以包含特定类型的所有实例的共享属性或者方法。这样,如果我们需要修改所有实例中的属性或者方法,就只需要修改一处,就能够影响到所有实例了。因为原型中的属性和方法是共享的。我们可以看下两个图示:

      构造函数方式

原型模式方式

  从上面的图示中我们就不难看出,为何下面的代码中"user1.show == user2.show;"返回的是ture,因为show方法是所有由User构造函数创建的对象所共享的,而不是每个对象都各自创建了一个show方法。

  每个JavaScript函数都有prototype属性,这个属性引用了一个对象,这个对象就是原型对象。原型对象初始化的时候是空的,我们可以在里面自定义任何属性和方法,这些方法和属性都将被该构造函数所创建的对象继承。

  在原型中添加属性和方法可以参照如下代码:

  1. function User(name,age){//构造方法
  2. this.name = name;//对象属性
  3. this.age = age;
  4. }
  5. User.prototype.addr = '湖北武汉';//在原型中添加属性
  6. User.prototype.show = function(){//在原型中添加方法
  7. alert(this.name+'|'+this.age);
  8. };
  9. var user1 = new User('ZXC',22);//创建实例
  10. var user2 = new User('CXZ',21);
  11. user1.show();//调用show()方法
  12. user2.show();
  13. alert(user1.show == user2.show);//返回 true 说明show方法是共享的
  14. alert(user1.addr);//'湖北武汉'
  15. alert(user2.addr);//'湖北武汉'

  但是有个问题是:如果我们既在构造方法中添加了一个属性、又在原型中添加了该属性,还在实例中添加了该属性,那么我们访问的究竟 是哪一个属性呢?我们先看看下面的代码:

  1. function User(name,age){//构造方法
  2. this.name = name;//对象属性
  3. this.age = age;
  4. this.addr = '湖北恩施';
  5. }
  6. User.prototype.addr = '湖北武汉';//在原型中添加属性
  7. var user1 = new User('ZXC',22);//创建实例
  8. var user2 = new User('CXZ',21);
  9. alert(user1.addr);//'湖北恩施'
  10. delete user1.addr;//删除对象属性
  11. alert(user1.addr);//'湖北武汉'
  12. delete User.prototype.addr;
  13. alert(user1.addr);//'undefined'
  14. user2.addr = '武汉';
  15. alert(user2.addr);//'武汉'

  从上面的代码可以看出,如果我们同时申明了对象属性、原型属性和实例属性,那么调用时显示的优先级应该是:实例属性>对象属性>原型属性。这就是采用了就近原则:调用时首先查找实例中是否直接定义了这个属性,有则返回实例属性;如果实例属性中没有就去构造函数中查找,有则返回;如果前面两者都没有就去原型对象中查找,如果没有则返回undefined。

  二 动态原型模式

  有人可能会觉得上面代码中的写法感觉很别扭,因为原型中的方法和属性与构造函数中定义的对象属性和方法不在一块儿,要是能封装在一起就更加直观,如果要解决这个问题,就要用到动态原型模式;

  1. //动态原型模式
  2. function User(name,age){//构造方法
  3. this.name = name;//属性
  4. this.age = age;
  5. this.addr = '湖北恩施';
  6. User.prototype.addr = '湖北武汉';//在原型中添加属性
  7. User.prototype.show = function(){//在原型中添加方法
  8. alert(this.name+'|'+this.age+'|'+this.addr);
  9. };
  10. }
  11. var user1 = new User('ZXC',22);//创建实例
  12. var user2 = new User('CXZ',21);
  13. user1.show();//调用show()方法
  14. user2.show();
  15. alert(user1.show==user2.show);//返回 true

  上面的代码看起来要更加直观。但是这样还是会有一些小的问题,就是我们在创建多个实例的时候,没创建一个实例就会在原型中重新创建一次原型中的方法。先来测试一下:

  1. alert('开始创建show……');
  2. User.prototype.show = function(){//在原型中添加方法
  3. alert(this.name+'|'+this.age+'|'+this.addr);
  4. };
  5. alert('结束创建show……');

  如果我们添加上面的alert(),运行时发现,没创建一个实例都会弹出两次对话框。这就证明了上面提到的重新创建的问题,虽然这样来说空间没有额外增加,但是时间却是增加了,因为每次都要重新创建。

  要解决这个问题,我们的思路是:首先判断show方法是否存在,如果不存在则创建,如果已经存在就不在重新创建。改进代码如下:

  1. if(this.show==undefined){//如果run方法还没有被创建
  2. alert('开始创建show……');
  3. User.prototype.show = function(){//在原型中添加方法
  4. alert(this.name+'|'+this.age+'|'+this.addr);
  5. };
  6. alert('结束创建show……');
  7. }

  运行发现,不管创建多少个实例都只会弹出两次对话框,这样就避免了不必要的开销。

  三 使用字面量方式创建原型

  除了上面提到的创建原型的方式,我们还可以用字面量方式创建,代码如下:

  1. //使用字面量方式创建原型
  2. function User(name,age){//构造方法
  3. this.name = name;//属性
  4. this.age = age;
  5. }
  6. User.prototype = {
  7. addr : '湖北武汉',
  8. show : function(){
  9. alert(this.name+'|'+this.age+'|'+this.addr);
  10. }
  11. };
  12. var user1 = new User('ZXC',22);//创建实例
  13. var user2 = new User('CXZ',21);
  14. user1.show();//调用show()方法
  15. user2.show();

  这里要说明的是:使用字面量方式创建后,不能再使用字面量的方式重写原型,一旦重写了原型,原来的原型中定义的所有属性和方法都将被清除。如下:

  1. //使用字面量方式创建原型
  2. function User(name,age){//构造方法
  3. this.name = name;//属性
  4. this.age = age;
  5. }
  6. User.prototype = {
  7. addr : '湖北武汉',
  8. show : function(){
  9. alert(this.name+'|'+this.age+'|'+this.addr);
  10. }
  11. };
  12. //重写了原型
  13. User.prototype = {
  14. other : '暂时没有说明……',
  15. show : function(){
  16. alert(this.addr);
  17. }
  18. };
  19. var user1 = new User('ZXC',22);//创建实例
  20. var user2 = new User('CXZ',21);
  21. user1.show();//返回 undefined
  22. user2.show();

  可见,一旦我们重写了原型,那么开始原型中定义的变量和方法都没有保存下来。但是,我们说的是不能用字面量的方式重写原型,那我们可不可以添加新的方法或者属性呢?答案是可以的,比如:

  1. User.prototype.addr1 = '武汉';

  这样就不会清楚原来原型中定义的方法和属性了。

  四 总结

  这里主要是介绍了原型的一些基本的知识,原型可以用来实现方法和属性的共享,也可以用来扩展对象的方法。比如我们可以用原型方法来扩展内置对象String的方法,让它具有VB的left()/right()功能,是实现截取字符串左边或者右边的功能。

  原型中还有一些其他的知识,这里没有涉及到,以后有时间继续讨论研究~~

Javascript中的对象和原型(3)的更多相关文章

  1. Javascript中的对象和原型(三)(转载)

    在Javascript中的对象和原型(二)中我们提到,用构造函数创建的对象里面,每个对象之间都是独立的,这样就会降低系统资源的利用率,解决这样问题,我们就要用到下面提到的原型对象. 一 原型对象 原型 ...

  2. Javascript中的对象和原型(一)(转载)

    面向对象的语言(如Java)中有类的概念,而通过类可以创建任意多个具有相同属性和方法的对象.但是,JavaScript 没有类的概念,因此它的对象也与基于类的语言中的对象有所不同. 要了解面向对象,首 ...

  3. javascript中的对象,原型,原型链和面向对象

    一.javascript中的属性.方法 1.首先,关于javascript中的函数/“方法”,说明两点: 1)如果访问的对象属性是一个函数,有些开发者容易认为该函数属于这个对象,因此把“属性访问”叫做 ...

  4. JavaScript中的对象与原型—你不知道的JavaScript上卷读书笔记(四)

    一.对象 对象可以通过两种形式定义:声明(文字)形式和构造形式.即: var myObj = { key: value // ... }; 或: var myObj = new Object(); m ...

  5. Javascript中的对象和原型

    一 原型对象 原型对象实际上就是构造函数的一个实例对象,和普通的实例对象没有本质上的区别.可以包含特定类型的所有实例的共享属性或者方法.这样,如果我们需要修改所有实例中的属性或者方法,就只需要修改一处 ...

  6. Javascript中的对象和原型(二)(转载)

    上一篇中提到了JavaScript中对象的创建的一些基本操作,接下来讨论下继续讨论. 一 工厂模式 我们知道,要创建一个对象我们可以用如下代码: var user = new Object(); // ...

  7. JavaScript内置对象与原型继承

    (一)   理解JavaScript类定义 1>关于内置对象理解 console.log(Date.prototype.__proto__===Object.prototype    //tru ...

  8. JavaScript中判断对象类型方法大全1

    我们知道,JavaScript中检测对象类型的运算符有:typeof.instanceof,还有对象的constructor属性: 1) typeof 运算符 typeof 是一元运算符,返回结果是一 ...

  9. (转)javascript中的对象查找

    本文转自:http://otakustay.com/object-lookup-in-javascript/  ---很棒的一篇文章,作者的其他文章还暂时没读,但相信作者是一个谦虚 谨慎的好工程师 近 ...

随机推荐

  1. Play 可以做的 5 件很酷的事

    Play 可以做的 5 件很酷的事 本章译者:@Playframwork 通过 5 个实例,透视 Play 框架背后的哲学. 绑定 HTTP 参数到 JAVA 方法参数 用 Play 框架,在 Jav ...

  2. Sublime Text3 快捷键汇总

    Ctrl+D 选词 (反复按快捷键,即可继续向下同时选中下一个相同的文本进行同时编辑)Ctrl+G 跳转到相应的行Ctrl+J 合并行(已选择需要合并的多行时)Ctrl+L 选择整行(按住-继续选择下 ...

  3. Android中进程与线程

    常说的主线程(UI线程)是什么? 当一个Android程序刚启动的时候,我们的android系统就会启动一个带有一个单一线程的linux进程.默认情况下,所有的组件比如Activity都运行在同样的一 ...

  4. Spring(八)SSH整合简述

    一.Spring与Struts2的整合 1.1.整合步骤 1.2.配置web.xml 1.3.配置spring配置文件applicationContext.xml 1.4.配置struts配置文件 1 ...

  5. git 查看远程分支、本地分支、删除本地分支

    1 查看远程分支 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 $ git branch -a   * br-2.1.2.2   master   remotes/origi ...

  6. iOS开发网络篇—搭建本地服务器(待整理)

      一.简单说明 说明:提前下载好相关软件,且安装目录最好安装在全英文路径下.如果路径有中文名,那么可能会出现一些莫名其妙的问题. 提示:提前准备好的软件 apache-tomcat-6.0.41.t ...

  7. centos性能监控系列二:Collectl初解

    对于一个 Linux 系统管理员来说确保自己管理的系统处于一个良好的状态是其首要责任. Linux 系统管理员可以找到有很多工具来帮助自己监控和显示系统中的进程,例如 top 和 htop 今天介绍一 ...

  8. Homebrew OS X 不可或缺的套件管理器

    Homebrew OS X 不可或缺的套件管理器,可以说Homebrew就是mac下的apt-get.yum. 1.安装homebrew brew的安装很简单,使用一条ruby命令即可,Mac系统上已 ...

  9. 【转载】改善数据质量从数据剖析(Data Profiling)开始

    市场研究公司Forrester副总裁Erin Kinikin曾经把低劣的数据质量做了一个形象的比喻“用更好的方法访问劣质的数据,结果类似于把已经腐烂了的桃子用更快的卡车,走更好的路线运输到达市场时,桃 ...

  10. Eclipse 一直提示 loading descriptor for 的解决方法

    启动eclipse之后,进行相关操作时,弹出界面,提示:loading descriptor for xxx 解决方法: 在Eclipse左侧的Project Explorer 最右上角有一个小钮,鼠 ...