深入理解JavaScript是如何实现继承的


Posted in Javascript onDecember 12, 2013

最近最网上看了一个人面试淘宝时的经历,然后发现了自己有好多好多不太清楚的地方,所以特此写点文章来加深自己对一些问题的理解。

文章中提到了一个问题是:JavaScript是如何实现继承的?

下面我便阐述一些在网上找到的方法和实例来解释下,借以加深自己的印象。

我们知道JavaScript中的function是万能的,除了用于的函数定义,也可以用于类的定义。

JavaScript的继承,说起来也是有点怪,不像C++和一些面向对象的语言,他没有public,private等访问控制修饰,也没有implement或其他特定的符号来说明是实现继承。

关于javascript类的继承可以参考一下下面的这个例子。

<script type="text/javascript"> 
function Person() {
    // 属性 
    this.Gender = "female";
    this.Age = 18;
    this.Words = "Silence";
    // 方法
    this.shouting = function() {
        alert("开心哦!父类的方法");
    }
}
// 继承
function Programmer() {
    this.base = Person;
}
Programmer.prototype = new Person;
// 为子类添加新的方法
Programmer.prototype.typeCode = function() {
    alert("俺是敲代码的!IT民工,很不开心。子类的方法");
}
// 调用示例
function sayHello() {
    var a = new Programmer();
    alert(a.Gender); // 调用父类的属性
    a.shouting(); // 调用父类的方法
    a.typeCode(); // 调用子类的方法
}        
sayHello();
</script>

上例中,首先是声明一个person类,里面包含了一些属性和方法,然后接着又声明了一个programmer类,其中有个base属性,这个属性并不是必需的,但是出于规范以及以后在查找对象所继承的类时都需要写上,然后是给programmer的原型对象(prototype)拷贝了person类;于是便实现了类的继承。

模拟JavaScript中类和继承的一些原理

在面向对象的语言中,我们使用类来创建一个自定义对象。然而JavaScript中所有事物都是对象,那么用什么办法来创建自定义对象呢?

这就需要引入另外一个概念 - 原型(prototype),我们可以简单的把prototype看做是一个模版,新创建的自定义对象都是这个模版(prototype)的一个拷贝 (实际上不是拷贝而是链接,只不过这种链接是不可见,给人们的感觉好像是拷贝)。

让我们看一下通过prototype创建自定义对象的一个例子:

// 构造函数
  function Person(name, sex) {
      this.name = name;
      this.sex = sex;
  }
  // 定义Person的原型,原型中的属性可以被自定义对象引用
  Person.prototype = {
      getName: function() {
          return this.name;
      },
      getSex: function() {
          return this.sex;
      }
  }

这里我们把函数Person称为构造函数,也就是创建自定义对象的函数。可以看出,JavaScript通过构造函数和原型的方式模拟实现了类的功能。

下面通过一个例子来具体阐述创建一个自定义对象,javascript所做的具体的工作:

var zhang = new Person("ZhangSan", "man");
console.log(zhang.getName()); // "ZhangSan"
var chun = new Person("ChunHua", "woman");
console.log(chun.getName()); // "ChunHua"

当代码var zhang = new Person("ZhangSan", "man")执行时,其实内部做了如下几件事情:

创建一个空白对象(new Object())。
拷贝Person.prototype中的属性(键值对)到这个空对象中(我们前面提到,内部实现时不是拷贝而是一个隐藏的链接)。
将这个对象通过this关键字传递到构造函数中并执行构造函数。
将这个对象赋值给变量zhang。
所有工作完成。
为了证明prototype模版并不是被拷贝到实例化的对象中,而是一种链接的方式,请看如下代码:

function Person(name, sex) {
    this.name = name;
    this.sex = sex;
}
Person.prototype.age = 20;
var zhang = new Person("ZhangSan", "man");
console.log(zhang.age); // 20
// 覆盖prototype中的age属性
zhang.age = 19;
console.log(zhang.age); // 19
delete zhang.age;
// 在删除实例属性age后,此属性值又从prototype中获取
console.log(zhang.age); // 20

在上面的这个例子中,如果他仅仅是通过拷贝得来的,则在删除了这个age这个属性后,这个对象里面将不会存在,但是例子中的age属性还能输出,还是覆盖以前的值,说明我们仅仅是删除了子类中同名的属性,而父类当中的age属性通过一种不可见的链接依然存在在对象中。

如何在JavaScript中实现简单的继承?

下面的例子将创建一个雇员类Employee,它从Person继承了原型prototype中的所有属性。

function Employee(name, sex, employeeID) {
    this.name = name;
    this.sex = sex;
    this.employeeID = employeeID;
}
// 将Employee的原型指向Person的一个实例
// 因为Person的实例可以调用Person原型中的方法, 所以Employee的实例也可以调用Person原型中的所有属性。
Employee.prototype = new Person();
Employee.prototype.getEmployeeID = function() {
    return this.employeeID;
};
var zhang = new Employee("ZhangSan", "man", "1234");
console.log(zhang.getName()); // "ZhangSan

好了,以上就是一些关于javascript实现继承的具体过程,和实现继承的方法。

当然总结一下,javascript中的继承机制仅仅是靠模拟的,于一些面向对象的语言来讲,显的粗糙而且还有一些缺陷,不过总的来讲,这依然不并会降低前端开发者在这方面的热情。

Javascript 相关文章推荐
移动端JQ插件hammer使用详解
Jul 03 Javascript
JavaScript调用浏览器打印功能实例分析
Jul 17 Javascript
js封装成插件_Canvas统计图插件编写实例
Sep 12 Javascript
Vue-resource拦截器判断token失效跳转的实例
Oct 27 Javascript
json前后端数据交互相关代码
Sep 19 Javascript
vue实现的下拉框功能示例
Jan 29 Javascript
详解JS实现系统登录页的登录和验证
Apr 29 Javascript
基于vue-cli3+typescript的tsx开发模板搭建过程分享
Feb 28 Javascript
前端深入理解Typescript泛型概念
Mar 09 Javascript
5个你不知道的JavaScript字符串处理库(小结)
Jun 01 Javascript
Jquery 获取相同NAME 或者id删除行操作
Aug 24 jQuery
vue实现省市区联动 element-china-area-data插件
Apr 22 Vue.js
Javascript Ajax异步读取RSS文档具体实现
Dec 12 #Javascript
javascripit实现密码强度检测代码分享
Dec 12 #Javascript
jQuery获取Radio,CheckBox选择的Value值(示例代码)
Dec 12 #Javascript
Jquery 改变radio/checkbox选中状态,获取选中的值(示例代码)
Dec 12 #Javascript
JavaScript判断访问的来源是手机还是电脑,用的哪种浏览器
Dec 12 #Javascript
js中typeof的用法汇总
Dec 12 #Javascript
使用js判断数组中是否包含某一元素(类似于php中的in_array())
Dec 12 #Javascript
You might like
PHP学习笔记 (1) 环境配置与代码调试
2011/06/19 PHP
PHP积分兑换接口实例
2015/02/09 PHP
phpcms手机内容页面添加上一篇和下一篇
2015/06/05 PHP
php实现数组纵向转横向并过滤重复值的方法分析
2017/05/29 PHP
php的对象传值与引用传值代码实例讲解
2021/02/26 PHP
javascript中字符串拼接需注意的问题
2010/07/13 Javascript
Jquery如何实现点击时高亮显示代码
2014/01/22 Javascript
javascript生成json数据简单示例分享
2014/02/14 Javascript
JavaScript实现鼠标滑过图片变换效果的方法
2015/04/16 Javascript
JavaScript forEach()遍历函数使用及介绍
2015/07/08 Javascript
详解js中call与apply关键字的作用
2016/11/21 Javascript
分享一道关于闭包、bind和this的面试题
2017/02/20 Javascript
详解webpack 多入口配置
2017/06/16 Javascript
JavaScript中最常用的10种代码简写技巧总结
2017/06/28 Javascript
js推箱子小游戏步骤代码解析
2018/01/10 Javascript
利用Console来Debug的10个高级技巧汇总
2018/03/26 Javascript
jQuery发请求传输中文参数乱码问题的解决方案
2018/05/22 jQuery
关于vue2强制刷新,解决页面不会重新渲染的问题
2019/10/29 Javascript
JS脚本实现定时到网站上签到/签退功能
2020/04/22 Javascript
详解JavaScript原型与原型链
2020/11/16 Javascript
JavaScript实现瀑布流布局的3种方式
2020/12/27 Javascript
[53:20]2018DOTA2亚洲邀请赛 4.1 小组赛 A组加赛 VG vs OG
2018/04/03 DOTA
[01:29:17]RNG vs Liquid 2019国际邀请赛淘汰赛 败者组 BO3 第二场 8.23
2019/09/05 DOTA
Python中类的继承代码实例
2014/10/28 Python
详解用python实现基本的学生管理系统(文件存储版)(python3)
2019/04/25 Python
python中pip的使用和修改下载源的方法
2019/07/08 Python
python实现简单颜色识别程序
2020/02/19 Python
pandas使用函数批量处理数据(map、apply、applymap)
2020/11/27 Python
深入浅析CSS3中的Flex布局整理
2020/04/27 HTML / CSS
贝玲妃美国官方网站:Benefit美国
2016/08/28 全球购物
药物学专业学生的自我评价
2013/10/27 职场文书
护士见习期自我鉴定
2014/02/08 职场文书
解除财产保全担保书
2014/05/20 职场文书
物业管理专业求职信
2014/06/11 职场文书
运动会主持词大全
2015/07/02 职场文书
详解CSS开发过程中的20个快速提升技巧
2021/05/21 HTML / CSS