zoukankan      html  css  js  c++  java
  • JavaScript是如何实现继承的

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

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

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

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

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

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

     1 <script type="text/javascript"> 
     2 function Person() {
     3     // 属性 
     4     this.Gender = "female";
     5     this.Age = 18;
     6     this.Words = "Silence";
     7     // 方法
     8     this.shouting = function() {
     9         alert("开心哦!父类的方法");
    10     }
    11 }
    12 // 继承
    13 function Programmer() {
    14     this.base = Person;
    15 }
    16 Programmer.prototype = new Person;
    17 // 为子类添加新的方法
    18 Programmer.prototype.typeCode = function() {
    19     alert("俺是敲代码的!IT民工,很不开心。子类的方法");
    20 }
    21 // 调用示例
    22 function sayHello() {
    23     var a = new Programmer();
    24     alert(a.Gender); // 调用父类的属性
    25     a.shouting(); // 调用父类的方法
    26     a.typeCode(); // 调用子类的方法
    27 }        
    28 sayHello();
    29 </script>

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

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

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

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

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

     1 // 构造函数
     2   function Person(name, sex) {
     3       this.name = name;
     4       this.sex = sex;
     5   }
     6   // 定义Person的原型,原型中的属性可以被自定义对象引用
     7   Person.prototype = {
     8       getName: function() {
     9           return this.name;
    10       },
    11       getSex: function() {
    12           return this.sex;
    13       }
    14   }

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

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

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

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

    1. 创建一个空白对象(new Object())。
    2. 拷贝Person.prototype中的属性(键值对)到这个空对象中(我们前面提到,内部实现时不是拷贝而是一个隐藏的链接)。
    3. 将这个对象通过this关键字传递到构造函数中并执行构造函数。
    4. 将这个对象赋值给变量zhang。
    5. 所有工作完成。

      为了证明prototype模版并不是被拷贝到实例化的对象中,而是一种链接的方式,请看如下代码:

     1 function Person(name, sex) {
     2     this.name = name;
     3     this.sex = sex;
     4 }
     5 Person.prototype.age = 20;
     6 var zhang = new Person("ZhangSan", "man");
     7 console.log(zhang.age); // 20
     8 // 覆盖prototype中的age属性
     9 zhang.age = 19;
    10 console.log(zhang.age); // 19
    11 delete zhang.age;
    12 // 在删除实例属性age后,此属性值又从prototype中获取
    13 console.log(zhang.age); // 20

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

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

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

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

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

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

      

      最后,第一次发博客,难免存在一些不足的地方,忘各位指正与见谅。

      另外推广下个人小站:www.xiangyifight.com

  • 相关阅读:
    win10 安装python教程
    nginx http请求无法加载https的css样式
    (第二十天)[js] 写一个验证身份证号的方法
    Linux重启nginx
    (第十一天)[js] 返回到顶部的方法有哪些?把其中一个方法出来
    看了一篇闭包的,推荐一下~
    HTTP状态码
    (第十天)[js] 写一个获取当前url查询字符串中的参数的方法
    (第九天)[js] 写一个判断数据类型的方法
    (第八天)[js] 写一个加密字符串的方法
  • 原文地址:https://www.cnblogs.com/xiangyi/p/3469587.html
Copyright © 2011-2022 走看看