JavaScript 原型、原型链与继承

本文深入解析JavaScript中的原型和原型链概念,探讨其如何实现继承,包括动态属性添加、构造函数与实例关系,以及ES6中class语法的底层原理。

JavaScript 不包含传统的类继承模型,而是使用 prototypal 原型模型。

虽然这经常被当作是 JavaScript 的缺点被提及,其实基于原型的继承模型比传统的类继承还要强大。实现传统的类继承模型是很简单,但是实现 JavaScript 中的原型继承则要困难的多。

由于 JavaScript 是唯一一个被广泛使用的基于原型继承的语言,所以理解两种继承模式的差异是需要一定时间的,今天我们就来了解一下原型和原型链。

var myObj = {};
function myFunc() {
}
console.log(myObj.prototype); // undefined
console.log(myFunc.prototype); // myFunc {}

通过 prototype 我们可以动态的向对象添加属性和方法,并且是可以继承的。

function myFunc(val) {
    this.val = val;
}

myFunc.prototype.getVal = function() {
        return this.val;
};
var myInstance = new myFunc(1);
// myInstance 调用原型属性
console.log(myInstance.getVal());
// 1

// 访问 myInstance 的原型

console.log(Object.getPrototypeOf(myInstance));
// myFunc { getVal: [Function] }

// 为 myInstance 添加属性 val
myInstance.val = 100;
// getVal 中的 this 指向 myInstance 实例
console.log(myInstance.getVal()); // 100

当继承的函数 getVal 被调用时, this 指向 myInstance 实例, 而不是原型 myFunc。

原型链

根据 ECMAScript 标准,someObject.[[Prototype]] 符号是用于指派 someObject 的原型。proto 属性(现已弃用)。

当一个对象访问一个属性时,查找的顺序是:

该对象自身属性 =》 该对象的原型的属性 =》 该对象的原型的原型属性 ... 直到找到匹配的属性或到达原型链末端(prototype 为 undefined)

var objA = function () {
    this.a = 'a';    this.b = 'b';
};
var myInstance = new objA();
myInstance.a = 'a in myInstance';
// myInstance 自身的属性有 a, 访问并停止查找属性

console.log(myInstance.a); // a in myInstance

// myInstance 自身的属性没有 b,
访问 myInstance 的原型 objA, objA 有属性 b, 访问并停止查找属性

console.log(myInstance.b); // b

// myInstance 自身和原型链上每个原型都没有属性 c,结果为 undefined

console.log(myInstance.c); // undefined

创建对象和原型链的几种方法

通过 JS 语法创建

var obj = {
    a: 1};
// 继承于 Object.prototype
// 原型链:obj => Object.prototype => undefined

console.log(Object.getPrototypeOf(obj));// {}

var arr = [1];
// 继承于 Array.prototype
// 原型链:arr => Array.prototype => undefined
console.log(Object.getPrototypeOf(arr)); // []

function func() {
}
// 继承于 Function.prototype
// 原型链:func => Function.prototype => undefined
console.log(Object.getPrototypeOf(func)); // [Function]

通过构造函数创建

在 JavaScript 中, 用 new 操作符来作用与一个函数时,这个函数就是构造函数 constructor。

var func = function() {
    this.val = 1;  
};

func.prototype.getVal = function() {
        return this.val;
};
var instance = new func();
// instance 是构造函数 func 的一个实例对象
// instance 自身具有 val 属性
// instance.[[Prototype]] 指向 func.prototype

通过 Object.create 创建

var a = {
    a: 'a'
};
var b = Object.create(a);
// b 继承于 a.prototype

// 原型链:b => a => Object.prototype => undefined

console.log(b.a) // a

通过 class 关键字 [ES6]

ES6 中的 class 只是基于原型继承的一种语法糖:

'use strict';
class Person {
    constructor(firstName, lastName) {
          this.firstName = firstName;
           this.lastName = lastName;
    }
}
class Student extends Person {
constructor(firstName, lastName) {
         super(firstName, lastName);
    }
    get name() {
         return this.firstName + ' ' + this.lastName;
    }
    set name(newFirstName) {
         this.firstName = newFirstName;
    }
}
var me = new Student('Dog', 'Dong');
console.log(me.name); // Dog Dong
me.name = 'Dog2';
console.log(me.name); // Dog2 Dong

其他

hasOwnProperty()

检测某个对象自身(不含原型链)是否含有某属性。

var objA = {
    valA: 'val a'
};
var objB = Object.create(objA);
objB.valB = 'val b';

console.log(objA.hasOwnProperty('valA')); // true
console.log(objB.hasOwnProperty('valA')); // false
console.log(objA.hasOwnProperty('valB')); // false
console.log(objB.hasOwnProperty('valB')); // true

在原型链上查找属性是很耗时的,尤其是访问一个不存在的属性时,整个原型链会被遍历。

Object.getPrototypeOf() [ES6]

返回对象的原型。

var objA = {
    valA: 'val a'

};
var objB = Object.create(objA);
console.log(Object.getPrototypeOf(objB)); 
// { valA: 'val a' }

Object.setPrototypeOf() [ES6]

设置对象的原型。

var objA = {
    valA: 'val a'};
var objB = {};
Object.setPrototypeOf(objB, objA);
console.log(Object.getPrototypeOf(objB)); 
// { valA: 'val a' }

需求响应动态冰蓄冷系统需求响应策略的优化研究(Matlab代码实现)内容概要:本文围绕需求响应动态冰蓄冷系统及其优化策略展开研究,结合Matlab代码实现,探讨了在电力需求侧管理背景下,冰蓄冷系统如何通过优化运行策略参需求响应,以实现削峰填谷、降低用电成本和提升能源利用效率的目标。研究内容包括系统建模、负荷预测、优化算法设计(如智能优化算法)以及多场景仿真验证,重点分析不同需求响应机制下系统的经济性和运行特性,并通过Matlab编程实现模型求解结果可视化,为实际工程应用提供理论支持和技术路径。; 适合人群:具备一定电力系统、能源工程或自动化背景的研究生、科研人员及从事综合能源系统优化工作的工程师;熟悉Matlab编程且对需求响应、储能优化等领域感兴趣的技术人员。; 使用场景及目标:①用于高校科研中关于冰蓄冷系统需求响应协同优化的课题研究;②支撑企业开展楼宇能源管理系统、智慧园区调度平台的设计仿真;③为政策制定者评估需求响应措施的有效性提供量化分析工具。; 阅读建议:建议读者结合文中Matlab代码逐段理解模型构建算法实现过程,重点关注目标函数设定、约束条件处理及优化结果分析部分,同时可拓展应用其他智能算法进行对比实验,加深对系统优化机制的理解。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值