因为之前谢过一篇关于原型对象的笔记:浅谈JavaScript中的原型模式。现在我又重新看到这个话题,对原型有了进一步的理解,所以,又要谈谈原型对象。

一、理解原型对象

创建的每一个函数都有一个prototype属性,它指向这个函数的原型对象。利用原型模式创建的方法和属性是被所有实例所共享的。

function Person(){}
Person.prototype.name="dwqs";
Person.prototype.age=20;
Person.prototype.sayName=function()
{
alert(this.name);
};
var per1 = new Person();
per1.sayName(); //dwqs
var per2 = new Person();
per2.sayName(); //dwqs
alert(per1.sayName == per2.sayName); //true

默认情况下,prototype指向的原型对象自动获取一个constructor属性,指向prototype属性所在的函数,而原型对象的其它方法和属性均从Object继承。

当调用Person的构造函数创建对象per1和per2时,Person的每个实例(per1和per2)均包含一个指向构造函数(Person)的原型对象的指针。ECMA-262第5版给指针取名为[[Prototype]]。

所以,实例中的指针仅指向原型,而不指向构造函数。虽然没有标准的方法直接访问[[Prototype]]特性,但在DOM笔记(十一)中提供了hasOwnProperty()和isPropertyOf()方法来反应原型对象和实例之间的关系。

//isPrototypeOf(obj)检测obj的[[prototype]]是否指向调用此方法的对象
alert(Person.prototype.isPrototypeOf(per1)); //true
alert(Person.prototype.isPrototypeOf(per2)); //true

hasOwnProperty()则是判断属性是存在原型中还是实例中,只有存在实例中时,才返回true。

per1.blog = "www.ido321.com";
alert(per1.hasOwnProperty("blog")); //true
alert(Person.prototype.hasOwnProperty("blog")); //false
alert(per1.hasOwnProperty("name")); //false
alert(Person.prototype.hasOwnProperty("name")); //true

若实例属性和原型属性同名,则同名的原型属性会被屏蔽,但是不会修改原型中的同名属性,当删除同名的实例属性后,又能重新访问被屏蔽的原型属性。

per1.name="i94web";
alert(per1.name); //i94web,来自实例
alert(per2.name); //dwqs 来自原型
delete per1.name;
alert(per1.name); //dwqs

ECMAScript 5提供了Object.getPrototypeOf(obj)来访问obj的[[Prototype]]的指向,即返回obj的原型对象

alert(Object.getPrototypeOf(per1) == Person.prototype);   //true
alert(Object.getPrototypeOf(per1).name);                 //dwqs

二、__proto__属性

在IE11、FireFox、Google等现代浏览器中,每个对象支持__proto__(前后都是两个下划线)属性用于访问父类的原型对象。但要注意的是:

1、原型对象应用__proto__属性时,返回当前类的父类的原型对象的引用;当实例对象应用__proto__属性时,返回当前实例所属类的原型对象的引用。

2、Object.prototype.__proto__返回null,Object没有父类。

3、prototype是静态属性,__proto__是实例属性。

//全是true
alert(Array.prototype.__proto__ == Object.prototype);
alert((new Array()).__proto__ == Array.prototype);
alert(Person.prototype.__proto__ == Object.prototype);
alert(per1.__proto__ == Person.prototype);
alert(Object.prototype.__proto__ == null);

三、重写原型

如果在原型上定义属性或者方法,每次都要将Person.prototype敲一遍,很麻烦。一个简单的方法就是重写原型。

function Person(){}
Person.prototype={
name:"dwqs",
age:20,
sayName:function()
{
alert(this.name);
}
};
var per1 = new Person();
alert(per1 instanceof Object); //true
alert(per1 instanceof Person); //true
alert(per1.constructor == Object); //true
alert(per1.constructor == Person); //false

因为重写了默认的prototype对象,所以constructor属性指向了新对象的constructor属性(Object构造函数),而不是指向Person了。如果constructor指向很重要,可以重新定义:

function Person(){}
Person.prototype={
constructor:Person, //重新定义constructor
name:"dwqs",
age:20,
sayName:function()
{
alert(this.name);
}
};
var per1 = new Person();
alert(per1.constructor == Person); //true

这样设置之后,constructor的特性[[Enumerable]]是true了(参考DOM笔记(十一)),即可以枚举,默认情况下,该属性不可枚举。

正如前文说的,实例中的指针仅指向原型,而不指向构造函数。所以重写原型对象之后,会切断现有原型和任何之前已经存在的对象实例之间的联系。

function Person(){}

//先创建一个实例
var per1 = new Person(); //重写原型
Person.prototype={
constructor:Person,
name:“dwqs”,
age:20,
sayName:function()
{
alert(this.name);
}
};
per1.sayName(); //错误:Uncaught TypeError: undefined is not a function

重写原型对象之前是这样的:

重写原型对象变成这样了:

四、原生对象的原型

所有原生引用类型,如Object、Array、String等都在其构造函数的原型上定义了方法。在Array.prototype上可以找到sort(),在String.prototype上可以找到substring

alert(typeof Array.prototype.sort);              //function
alert(typeof String.prototype.substring); //function

因此,也可以在原型对象上为引用类型定义新的方法

String.prototype.startWith=function(str)
{
return this.indexOf(str) == 0;
}
var msg = "Hello world";
alert(msg.startWith("Hello")); //true

原文首发:http://www.ido321.com/1372.html

DOM笔记(十二):又谈原型对象的更多相关文章

  1. python cookbook第三版学习笔记十二:类和对象(三)创建新的类或实例属性

    先介绍几个类中的应用__getattr__,__setattr__,__get__,__set__,__getattribute__,. __getattr__:当在类中找不到attribute的时候 ...

  2. 《C++游戏开发》笔记十二 战争迷雾:初步实现

    本系列文章由七十一雾央编写,转载请注明出处.  http://blog.csdn.net/u011371356/article/details/9475979 作者:七十一雾央 新浪微博:http:/ ...

  3. python3.4学习笔记(十二) python正则表达式的使用,使用pyspider匹配输出带.html结尾的URL

    python3.4学习笔记(十二) python正则表达式的使用,使用pyspider匹配输出带.html结尾的URL实战例子:使用pyspider匹配输出带.html结尾的URL:@config(a ...

  4. Go语言学习笔记十二: 范围(Range)

    Go语言学习笔记十二: 范围(Range) rang这个关键字主要用来遍历数组,切片,通道或Map.在数组和切片中返回索引值,在Map中返回key. 这个特别像python的方式.不过写法上比较怪异使 ...

  5. DirectX11笔记(十二)--Direct3D渲染8--EFFECTS

    原文:DirectX11笔记(十二)--Direct3D渲染8--EFFECTS 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/u010333737 ...

  6. java jvm学习笔记十二(访问控制器的栈校验机制)

    欢迎装载请说明出处:http://blog.csdn.net/yfqnihao 本节源码:http://download.csdn.net/detail/yfqnihao/4863854 这一节,我们 ...

  7. DOM笔记(十一):JavaScript对象的基本认识和创建

    一.什么是对象? 面 向对象(Object-Oriented,OO)的语言有一个标志,那就是都有类的概念,例如C++.Java等:但是ECMAScript没有类的概 念.ECMAScript-262把 ...

  8. JavaScript权威设计--命名空间,函数,闭包(简要学习笔记十二)

    1.作为命名空间的函数 有时候我们需要声明很多变量.这样的变量会污染全局变量并且可能与别人声明的变量产生冲突. 这时.解决办法是将代码放入一个函数中,然后调用这个函数.这样全局变量就变成了 局部变量. ...

  9. DOM笔记(二):Node接口

    所有的节点都使用Node接口来表示,可以使用很多方法去获取节点,如document.getElementsByTagName().document.getElementsByName()等均返回一个N ...

随机推荐

  1. Java 初始化和清理

    初始化和清理是影响代码安全的两个重要因素. 一.初始化 1. 方法重载 构造器与类名相同,成为强制重载方法名的原因之一.重载规则:每个重载的方法必须拥有独一无二的参数类型列表.不能根据返回值来区分重载 ...

  2. 2.Spring的Bean生命周期和组装方式

    1.Spring IoC容器概述 Spring IoC容器: Spring容器即体现了IoC原理    Spring容器通过读取配置元数据负责对Beans实例化.配置和装配     配置元数据可以用X ...

  3. Oozie的缺点

    Oozie使用的时候有以下不便: [a]Oozie调度的Workflow只能使用XML文件配置 [b]启动调度只能通过命令行 [c]无法通过Oozie界面调试调度脚本 [d]Oozie无法可视化调试脚 ...

  4. 多数据源 + Configuration中bean依赖注入顺序问题

    为什么要调用方法,而不是直接autowire? 官方文档 https://docs.spring.io/spring-boot/docs/current/reference/html/howto-da ...

  5. mgo03_linux7上安装mongo4.0

    下载地址https://www.mongodb.com/download-center#community tar -xvf mongodb-linux-x86_64-rhel70-4.0.0.tgz ...

  6. 安装tomcat时遇到的问题

    1.刚开始在eclipse配置的tomcat是免安装的,后来提示 所以后来配置了一个安装版的. 2.后来运行server发现报错:8080,8005,端口被占用,然后关闭xammp上的server,然 ...

  7. [转]使用jquery dataTable

    本文转自:http://blog.csdn.net/llhwin2010/article/details/8663753 jQuery 的插件 dataTables 是一个优秀的表格插件,提供了针对表 ...

  8. Clouder Manager安装时出现please remove the following file: /etc/cloudera-scm-server/db.properties问题解决(图文详解)

    问题详情 bigdata@ubuntucmbigdata1:/opt/softwares/cm-$ sudo ./cloudera-manager-installer.bin This install ...

  9. stm32 PWM输出学习

    STM32 的定时器除了 TIM6 和 7,其他的定时器都可以用来产生 PWM 输出.其中高级定时器 TIM1 和 TIM8 可以同时产生多达 7 路的 PWM 输出.通用定时器也能同时产生多达 4路 ...

  10. 关于GPU的 MAKEFILE

    引言 最近由于更换项目,服务器也被换走,估计一时半会用不到GPU了,因此最近想把前一段时间做的一些工作,整理记录一下. 实验室采用的GPU有两款: 1. 服务器上的板卡:NVIDIA的Tesla K2 ...