javascript学习笔记(九)javascript中的原型(prototype)及原型链的继承方式


Posted in Javascript onApril 12, 2011

在使用面向对象编程时,对象间的继承关系自然少不了!而原型正是实现javascript继承的很重要的一种方法!
我们首先来看以下代码:

function person(name, age) { 
this.name = name; 
this.age = age; 
} 
person.prototype.getInfo = function() { 
alert("My name is "+this.name+", and I have "+this.age+" years old"); 
} 
var zhangchen = new person("zhangchen", 23); 
zhangchen.getInfo(); //output My name is zhangchen, and I have 23 years old;

从运行的结果我们可以看出,通过关键字new创建的zhangchen这个对象继承了person中通过原型定义的getInfo()方法。下面我们具体来看新建的zhangchen这个对象是如何继承person对象的属性和方法的。
原型:在使用 JavaScript 的面向对象编程中,原型对象是个核心概念。在 JavaScript 中对象是作为现有示例(即原型)对象的副本而创建的,该名称就来自于这一概念。此原型对象的任何属性和方法都将显示为从原型的构造函数创建的对象的属性和方法。可以说,这些对象从其原型继承了属性和方法。当创建zhangchen对象时:
var zhangchen = new company("zhangchen", 23);

zhangchen 所引用的对象将从它的原型继承属性和方法,对象 zhangchen 的原型来自构造函数(在这里是函数 person)的属性。

在 JavaScript 中,每个函数都有名为prototype的属性,用于引用原型对象。此原型对象又有名为constructor的属性,它反过来引用函数本身。这是一种循环引用, 下图更好地说明了这种循环关系。

javascript学习笔记(九)javascript中的原型(prototype)及原型链的继承方式

图1 循环关系

现在,通过new运算符用函数(上面示例中为 person)创建对象时,所获得的对象将继承 person.prototype 的属性。在上图,可以看到 person.prototype 对象有一个回指 person 函数的构造函数属性。这样,每个 person对象(从 person.prototype 继承而来)都有一个回指 person 函数的构造函数属性。

我们可以用以下代码来验证这种循环是否正确:

function person(name, age) { 
this.name = name; 
this.age = age; 
} person.prototype.getInfo = function() { 
alert("My name is "+this.name+", and I have "+this.age+" years old"); 
} 
var zhangchen = new person("zhangchen", 23); 
alert(zhangchen.constructor == person.prototype.constructor); //output true 
alert(zhangchen.constructor == person);// output true 
alert(person.prototype.isPrototypeOf(zhangchen)); //output true

以上代码中的对"isPrototypeOf()"方法的调用来自哪里呢?是来自person.prototype对象吗?不对,实际上,在 person.prototype 和 person 实例中还可以调用其他方法,比如 toString、toLocaleString 和 valueOf,但它们都不来自 person.prototype。而是来自于JavaScript 中的 Object.prototype ,它是所有原型的最终基础原型。(Object.prototype 的原型是 null。)

在以上例中,zhangchen.prototype 是对象。它是通过调用 Object 构造函数创建的(尽管它不可见)相当于执行子以下代码:

zhangchen.prototype = new Object();

因此,正如 person实例继承person.prototype 一样,zhangchen.prototype 继承 Object.prototype。这使得所有 zhangchen 实例也继承了 Object.prototype 的方法和属性。

原型链:每个 JavaScript 对象都继承一个原型链,而所有原型都终止于 Object.prototype。注意,这种继承是活动对象之间的继承。它不同于继承的常见概念,后者是指在声明类时类之间的发生的继承。因此,JavaScript 继承动态性更强。它使用简单算法实现这一点,如下所示:当您尝试访问对象的属性/方法时,JavaScript 将检查该属性/方法是否是在该对象中定义的。如果不是,则检查对象的原型。如果还不是,则检查该对象的原型的原型,如此继续,一直检查到 Object.prototype。下图说明了此解析过程:

javascript学习笔记(九)javascript中的原型(prototype)及原型链的继承方式
图2 toString()方法的解析过程

从以上解析过程中,如果在对象中定义了属性/方法 X,则该对象的原型中将隐藏同名的属性/方法。例如,通过在 person.prototype 中定义 toString 方法,可以改写 Object.prototype 的 toString 方法。

再看以下代码:

function person(name, age) { 
this.name = name; 
this.age = age; 
} 
person.prototype.getInfo = function() { 
alert("My name is "+this.name+", and I have "+this.age+" years old"); 
} 
var zhangchen = new person("zhangchen", 23); 
var luomi = new person("luomi", 23); 
zhangchen.getInfo(); // output My name is zhangchen, and I have 23 years old; 
luomi.getInfo(); // output My name is luomi, and I have 23 years old; 
luomi.getInfo = function() { 
alert("here can rewrite the function of getInfo!"); 
} 
luomi.getInfo(); //here can rewrite the function of getInfo! 
zhangchen.getInfo(); // output My name is zhangchen, and I have 23 years old;

从运行结果可以看到,虽然person的每个实例都继承了person.prototype中的方法,但是我们也可以在实例化的对象中重新定义原型对象中的方法,而且也不会影响到其它的实例!

以上是自己对原型及原型链继承方式的认识,参考( JavaScript: 使用面向对象的技术创建高级 Web 应用程序),希望大家共同讨论!

Javascript 相关文章推荐
Locate a File Using a File Open Dialog Box
Jun 18 Javascript
jquery 插件学习(三)
Aug 06 Javascript
javascript/jquery获取地址栏url参数的方法
Mar 05 Javascript
JavaScript基础函数整理汇总
Jan 30 Javascript
jQuery 1.9.1源码分析系列(十五)之动画处理
Dec 03 Javascript
微信小程序实现带刻度尺滑块功能
Mar 29 Javascript
node通过npm写一个cli命令行工具
Oct 12 Javascript
详解基于 Nuxt 的 Vue.js 服务端渲染实践
Oct 24 Javascript
AngularJS实现的根据数量与单价计算总价功能示例
Dec 26 Javascript
使用mpvue搭建一个初始小程序及项目配置方法
Dec 03 Javascript
微信小程序picker组件关于objectArray数据类型的绑定方法
Mar 13 Javascript
详解 微信小程序开发框架(MINA)
May 17 Javascript
javascript 学习笔记(八)javascript对象
Apr 12 #Javascript
jQuery的初始化与对象构建之浅析
Apr 12 #Javascript
避免回车键导致的页面无意义刷新的解决方法
Apr 12 #Javascript
基于jquery实现的上传图片及图片大小验证、图片预览效果代码
Apr 12 #Javascript
javascript实现上传图片并预览的效果实现代码
Apr 11 #Javascript
window.dialogArguments 使用说明
Apr 11 #Javascript
30个最佳jQuery Lightbox效果插件分享
Apr 11 #Javascript
You might like
用PHP程序实现支持页面后退的两种方法
2008/06/30 PHP
深入Nginx + PHP 缓存详解
2013/07/11 PHP
PHP正则提取不包含指定网址的图片地址的例子
2014/04/21 PHP
兼容各大浏览器带关闭按钮的漂浮多组图片广告代码
2014/06/05 PHP
jQuery+jqmodal弹出窗口实现代码分明
2010/06/14 Javascript
jQuery 菜单随滚条改为以定位方式(固定要浏览器顶部)
2012/05/24 Javascript
firefox下jquery iframe刷新页面提示会导致重复之前动作
2012/12/17 Javascript
jquery 按键盘上的enter事件
2014/05/11 Javascript
jQuery实现固定在网页顶部的菜单效果代码
2015/09/02 Javascript
如何解决IONIC页面底部被遮住无法向上滚动问题
2016/09/06 Javascript
vue.js 1.x与2.0中js实时监听input值的变化
2017/03/15 Javascript
Vuex之理解Getters的用法实例
2017/04/19 Javascript
浅谈react受控组件与非受控组件(小结)
2018/02/09 Javascript
微信小程序使用map组件实现路线规划功能示例
2019/01/22 Javascript
vue.js 实现a标签href里添加参数
2019/11/12 Javascript
Vue开发中常见的套路和技巧总结
2020/11/24 Vue.js
python网络编程调用recv函数完整接收数据的三种方法
2017/03/31 Python
Python建立Map写Excel表实例解析
2018/01/17 Python
Python 读取指定文件夹下的所有图像方法
2018/04/27 Python
利用python循环创建多个文件的方法
2018/10/25 Python
python连接PostgreSQL过程解析
2020/02/09 Python
Python图像处理库PIL的ImageEnhance模块使用介绍
2020/02/26 Python
Python定义一个Actor任务
2020/07/29 Python
css3实现背景图片拉伸效果像桌面壁纸一样
2013/08/19 HTML / CSS
canvas绘制树形结构可视图形的实现
2020/04/03 HTML / CSS
ivx平台开发之不用代码实现一个九宫格抽奖功能
2021/01/27 HTML / CSS
html5使用window.postMessage进行跨域实现数据交互的一次实战
2021/02/24 HTML / CSS
请说出几个常用的异常类
2013/01/08 面试题
中学生打架检讨书
2014/02/10 职场文书
《赶海》教学反思
2014/04/20 职场文书
2015年依法行政工作总结
2015/04/29 职场文书
励志语录:你若不勇敢,谁替你坚强
2019/11/08 职场文书
vue实现锚点定位功能
2021/06/29 Vue.js
Java Spring Boot 正确读取配置文件中的属性的值
2022/04/20 Java/Android
win10蓝屏0xc0000001安全模式进不了怎么办?win10出现0xc0000001的解决方法
2022/08/05 数码科技
CSS 鼠标选中文字后改变背景色的实现代码
2023/05/21 HTML / CSS