[JavaStript学习记录] 对象的继承

news/2024/7/19 13:29:59 标签: prototype, javascript, js, 前端

---- 本系列仅为个人学习总结,基于阮一峰大佬的《>>JavaScript 教程<<》---

目录

一.原型对象

prototypep%E5%B1%9E%E6%80%A7-toc" style="margin-left:40px;">prototypep属性

 Constructor属性

二.instanceof 运算符 

注意:instanceof只能用于对象,不适用原始类型的值

三.构造函数的继承

第一步:

第二步:

另外一种写法(不推荐)

例如(让Rectangle构造函数继承Shape)

四.多重继承

★五.模块

loading......


一.原型对象

原型对象的作用,就是定义所有实例对象共享的属性和方法。这也是它被称为原型对象的原因,而实例对象可以视作从原型对象衍生出来的子对象。

prototypep%E5%B1%9E%E6%80%A7">prototypep属性

JavaScript 规定,每个函数都有一个prototype属性,指向一个对象。对于普通函数来说,该属性基本无用。但是,对于构造函数来说,生成实例的时候,该属性会自动成为实例对象的原型。

javascript">function Animal(name) {
  this.name = name;
}
Animal.prototype.color = 'white';

var cat1 = new Animal('玛卡巴卡');
var cat2 = new Animal('唔西迪西');

cat1.color //white
cat2.color //white

cat1.color = 'blue'

cat1.color //blue
cat2.color //white

读取对象的某个属性时,JavaScript 引擎先寻找对象本身的属性,如果找不到,就到它的原型去找,如果还是找不到,就到原型的原型去找。如果直到最顶层的Object.prototype还是找不到,则返回undefined。如果对象自身和它的原型,都定义了一个同名属性,那么优先读取对象自身的属性,这叫做“覆盖”(overriding)。

注意,一级级向上,在整个原型链上寻找某个属性,对性能是有影响的。所寻找的属性在越上层的原型对象,对性能的影响越大。如果寻找某个不存在的属性,将会遍历整个原型链。

javascript">var MyArray = function () {};

MyArray.prototype = new Array();

var mine = new MyArray();
mine.push(1, 2, 3);
mine.length // 3
mine instanceof Array // true

 Constructor属性

表示原型对象与构造函数之间的关联关系

  • 可以得知某个实例对象,到底是哪一个构造函数产生的
javascript">function F() {};
var f = new F();

f.constructor === F // true
f.constructor === G // false
  • 可以从一个实例对象新建另一个实例(间接调用构造函数)
javascript">function Constr() {}
var x = new Constr();

var y = new x.constructor();
y instanceof Constr // true

var m = new Constr();
m.constructor() == y.constructor()  //true

 修改原型对象时,一般要同时修改constructor属性的指向,防止引用的时候出错

javascript">//未修改constructor属性

function Person(name) {
  this.name = name;
}

Person.prototype.constructor === Person // true

Person.prototype = {
  method: function () {}
};

Person.prototype.constructor === Person // false
Person.prototype.constructor === Object // true
javascript">// 坏的写法
C.prototype = {
  method1: function (...) { ... },
  // ...
};

// 好的写法 ,在原型对象上添加方法
C.prototype = {
  constructor: C,
  method1: function (...) { ... },
  // ...
};

// 更好的写法
C.prototype.method1 = function (...) { ... };

二.instanceof 运算符 

返回一个布尔值,表示对象是否为某个构造函数的实例

javascript">var v = new Vehicle();
v instanceof Vehicle // true
Vehicle.prototype.isPrototypeOf(v) //true

由于任意对象(除了null)都是Object的实例,所以instanceof运算符可以判断一个值是否为非null的对象

javascript">var obj = { foo: 123 };
obj instanceof Object // true
null instanceof Object // false

instanceof的原理是检查右边构造函数的prototype属性,是否在左边对象的原型链上 

javascript">var obj = Object.create(null);
typeof obj // "object"
obj instanceof Object // false

上面代码中,Object.create(null)返回一个新对象obj,它的原型是nullObject.create()的详细介绍见后文)。右边的构造函数Objectprototype属性,不在左边的原型链上,因此instanceof就认为obj不是Object的实例。这是唯一的instanceof运算符判断会失真的情况(一个对象的原型是null)。

注意:instanceof只能用于对象,不适用原始类型的值

javascript">var s = 'hello';
s instanceof String // false

上面代码中,字符串不是String对象的实例(因为字符串不是对象),所以返回false

此外,对于undefinednullinstanceof运算符总是返回false

javascript">undefined instanceof Object // false
null instanceof Object // false

还可以巧妙地解决,调用构造函数时,忘了加new命令的问题

javascript">function Animal (age, color) {
  if (this instanceof Animal) {
    this._age = age;
    this._color = bar;
  } else {
    return new Animal (age, color);
  }
}

三.构造函数的继承

第一步:

在子类的构造函数中,调用父类的构造函数

javascript">function Sub(value) {
  Super.call(this); //★
  this.prop = value;
}

Sub 是子类的构造函数,this是子类的实例。
在实例上调用父类的构造函数Super,使子类实例具有父类实例的属性

第二步:

让子类的原型指向父类的原型,这样子类就可以继承父类原型

javascript">Sub.prototype = Object.create(Super.prototype);
Sub.prototype.constructor = Sub;  //修改constructor属性的指向,上文中有解释
Sub.prototype.method = '...';

 ★★★Sub.prototype是子类的原型,要将它赋值为Object.create(Super.prototype),而不是直接等于Super.prototype

否则后面两行对Sub.prototype的操作,会连父类的原型Super.prototype一起修改掉。

另外一种写法(不推荐)

(#^.^#)  老父亲也是需要隐私的哦!

 

javascript">Sub.prototype = new Super();

也有继承的效果,但是子类会具有父类实例的方法。有时,这可能不是我们需要的,所以不推荐这种写法

例如(Rectangle构造函数继承Shape)

javascript">function Shape() {
  this.x = 0;
  this.y = 0;
}

Shape.prototype.move = function (x, y) {
  this.x += x;
  this.y += y;
  console.info('Shape moved.');
};

// 第一步,子类继承父类的实例
function Rectangle() {
  Shape.call(this); // 调用父类构造函数
}
// 另一种写法
function Rectangle() {
  this.base = Shape;
  this.base();
}

// 第二步,子类继承父类的原型
Rectangle.prototype = Object.create(Shape.prototype);
Rectangle.prototype.constructor = Rectangle;


var rect = new Rectangle();

rect instanceof Rectangle  // true
rect instanceof Shape  // true

 在这个例子中,子类是整体继承父类。有时只需要单个方法的继承,这时可以采用下面的写法。

javascript">ClassB.prototype.print = function() {
  ClassA.prototype.print.call(this);
  // some code
}

子类B的print方法先调用父类A的print方法,
再部署自己的代码。这就等于继承了父类A的print方法

四.多重继承

 avaScript 不提供多重继承(即一个对象同时继承多个对象)功能。但是,可以通过变通方法,实现这个功能

下面的子类AB同时继承了父类AB。这种模式又称为 Mixin(混入)

javascript">function A(){
    this.h='hello';
}
function B(){
    this.w='world';
}
function AB(){
    A.call(this);
    B.call(this);
}

// 继承 A
AB.prototype = Object.create(A.prototype);
// 继承链中加入 B
Object.assign(AB.prototype,B.pprototype);
// 指定构造函数
AB.prototype.constructor = AB;


var one = new AB();
one.h    // 'hello'
one.w    // 'world'

★五.模块

loading......


http://www.niftyadmin.cn/n/816857.html

相关文章

[JavaStript学习记录] 对原型链的一些理解

回忆一下昨天的学习内容,梳理一些疑惑 先写了一个简单的构造函数 function A(){var num0; }var aanew A();查看该函数的实例 为什么实例对象没有prototype属性呢? 下面是经过查证后我的理解 发现 1.aa 是 A 的一个实例,但是不是一个函数&#xff0c;所以没有prototype; A是…

[JavaStript学习记录] 模块/封装私有变量

模块(用软件工程的方法&#xff0c;管理网页的业务逻辑) "立刻执行函数" 为了不暴露私有成员,使用"立刻执行函数",将有关的属性、方法封装在一个函数的作用域中 var m (function () {var count 0;var f1 function () {//...};var f2 function () {/…

2018-2019-1 20165333 《信息安全系统设计基础》第六周学习总结

2018-2019-1 20165333 《信息安全系统设计基础》第六周学习总结 教材内容总结 Unix I/O 打开文件。一个应用程序通过要求内核打开相应的文件&#xff0c;来宣告它想要访问一个I/O设备。Linux shell创建的每个进程开始都有三个打开的文件&#xff1a;标准输入&#xff08;0&…

[JavaScript学习记录] 首次运用于网页,做一个简易利息计算器!!!

目录 事件 错误处理 1.隐藏/显示文字 2.简单的加法计算器 效果: ★★★3.简易利息计算器 效果: ​​​​​​​ 事件 javascript允许html与用户交互的行为 例如:鼠标点击事件 <script> function sayHello{alert(Hello World ppp!); } </script><button …

ubuntu 手动修改分辨率为1920 X 1080 的两种方式

方案一&#xff08;临时性&#xff0c;重启会失效&#xff09;&#xff1a; 1、打开终端。输入&#xff1a;cvt 1920 1080 出现有modeline 的提示。 2、使用 xrandr 创建新的 mode&#xff0c;--newmode 后面跟刚刚 cvt 产生的 modeline 后面的信息&#xff0c;如果去掉“_60.0…

[JavaScript学习记录] BOM小记

浏览器对象模型(Browser Object Model) 浏览器对象包括 Window(窗口) Navigator(浏览器) Screen (客户端屏幕) History(访问历史) Location(浏览器地址) 目录 History Nevigator 弹框 警告框alert​ 确认框conform 输入框 History history.go(-2); //-1表示上次&#xf…

模板:缩点

翻之前记录翻出来缩点的板子 正好今天语文课看tarjan睡着了…… https://www.luogu.org/problemnew/show/P3387 1 #include<iostream>2 #include<cstdio>3 #include<queue>4 using namespace std;5 const int maxn1000010;6 int n, m, cnt, num, s, top;7 in…

[JavaScript学习记录] 异步操作

JavaScript 引擎有多个线程&#xff0c;单个脚本只能在主线程运行,其他线程都在后台配合. 单线程模型虽然对 JavaScript 构成了很大的限制&#xff0c;但也因此使它具备了其他语言不具备的优势。如果用得好&#xff0c;JavaScript 程序是不会出现堵塞的&#xff0c;这就是为什…