JS继承的几种方式

JS继承的几种方式

(1) 属性拷贝

就是将对象的成员复制一份给需要继承的对象
// 创建父对象
var superObj = {
  name: 'Li',
  age: 25,
  friends: ['小明', '小李', '小赵'],
  showName: function(){
    alert(this.name);
  }
}

// 创建需要继承的子对象
var subObj = {};

// 开始拷贝属性(使用for...in...循环)
for( var i in superObj ){
  subObj[i] = superObj[i];
}

console.log(subObj)
console.log(superObj)

存在问题:

如果继承过来的成员是引用类型的话,
那么这个引用类型的成员在父对象和子对象之间是共享的,
也就是说修改了之后, 父子对象都会受到影响.

(2) 原型式继承

借用构造函数的原型对象实现继承
// 创建父构造函数
function SuperClass(name){
  this.name = name;
  this.showName = function(){
    alert(this.name);
  }
}

// 设置父构造器的原型对象
SuperClass.prototype.showAge = function(){
  console.log(this.age);
}

// 创建子构造函数
function SubClass(){

}

// 设置子构造函数的原型对象实现继承
SubClass.prototype = SuperClass.prototype;

var child = new SubClass()

问题:

  1. 父构造函数的原型对象和子构造函数的原型对象上的成员有共享问题
  2. 只能继承父构造函数的原型对象上的成员, 不能继承父构造函数的实例对象的成员

(3) 原型链继承

即 子构造函数.prototype = new 父构造函数()
// 创建父构造函数
function SuperClass(){
    this.name = 'liyajie';
    this.age = 25;
    this.showName = function(){
        console.log(this.name);
    }
}
// 设置父构造函数的原型
SuperClass.prototype.friends = ['小名', '小强'];
SuperClass.prototype.showAge = function(){
    console.log(this.age);
}
// 创建子构造函数
function SubClass(){

}
// 实现继承
SubClass.prototype = new SuperClass();
// 修改子构造函数的原型的构造器属性
SubClass.prototype.constructor = SubClass;

var child = new SubClass();
console.log(child.name); // liyajie
console.log(child.age);// 25
child.showName();// liyajie
child.showAge();// 25
console.log(child.friends); // ['小名','小强']

// 当我们改变friends的时候, 父构造函数的原型对象的也会变化
child.friends.push('小王八');
console.log(child.friends);["小名", "小强", "小王八"]
var father = new SuperClass();
console.log(father.friends);["小名", "小强", "小王八"]

问题:

不能给父构造函数传递参数,父子构造函数的原型对象之间有共享问题

(4) 借用构造函数

使用call和apply借用其他构造函数的成员, 可以解决给父构造函数传递参数的问题, 但是获取不到父构造函数原型上的成员.也不存在共享问题
// 创建父构造函数
function Person(name){
  this.name = name;
  this.freinds = ['小王', '小强'];
  this.showName = function(){
     console.log(this.name);
  }
}

// 创建子构造函数
 function Student(name){
  // 使用call借用Person的构造函数
  Person.call(this, name);
 }

 // 测试是否有了 Person 的成员
 var stu = new Student('Li');
 stu.showName(); // Li
 console.log(stu.friends); // ['小王','小强']

(5) 组合继承

 借用构造函数 + 原型式继承
// 创建父构造函数
function Person(name,age){
    this.name = name;
    this.age = age;
    this.showName = function(){
        console.log(this.name);
    }
}
// 设置父构造函数的原型对象
Person.prototype.showAge = function(){
    console.log(this.age);
}
// 创建子构造函数
function Student(name){
    Person.call(this,name);
}
// 设置继承
Student.prototype = Person.prototype;
Student.prototype.constructor = Student;

上面代码解决了 父构造函数的属性继承到了子构造函数的实例对象上了,
并且继承了父构造函数原型对象上的成员
解决了给父构造函数传递参数问题

问题

共享的问题

(6) 借用构造函数 + 深拷贝

function Person(name,age){
    this.name = name;
    this.age = age;
    this.showName = function(){
        console.log(this.name);
    }
}
Person.prototype.friends = ['小王','小强','小王八'];

function Student(name,25){
    // 借用构造函数(Person)
    Person.call(this,name,25);
}
// 使用深拷贝实现继承
deepCopy(Student.prototype,Person.prototype);
Student.prototype.constructor = Student;

// 这样就将Person的原型对象上的成员拷贝到了Student的原型上了, 这种方式没有属性共享的问题.

 

深拷贝(递归)

使用递归实现, 主要是为了解决对象中引用类型的成员的共享问题.

好处是不管是值类型的属性还是引用类型的成员都不会有共享问题.

// 将obj2的成员拷贝到obj1中, 只拷贝实例成员
function deepCopy(obj1, obj2) {
    for (var key in obj2) {
        // 判断是否是obj2上的实例成员
        if (obj2.hasOwnProperty(key)) {
            // 判断是否是引用类型的成员变量
            if (typeof obj2[key] == 'object') {
                obj1[key] = Array.isArray(obj2[key]) ? [] : {};
                deepCopy(obj1[key], obj2[key]);
            } else {
                obj1[key] = obj2[key];
            }
        }
    }
}

var person = {
    name: 'liyajie',
    age: 25,
    showName: function() {
        console.log(this.name);
    },
    friends: [1, 2, 3, 4],
    family: {
        father: 'ligang',
        mather: 'sizhongzhen',
        wife: 'dan',
        baby: 'weijun'
    }
}
var student = {};
// 将person的成员拷贝到student对象上.
deepCopy(student, person);

Array.isArray()

此方法主要是来判断某个对象是否是数组, 因为是ES5的新特性, 所有有兼容性问题.

// 检查是否是数组对象
function checkIsArray(obj){
    if(Array.isArray){// 如果有这个属性
        return Array.isArray(obj);
    } else {
        return Object.prototype.toString.call(obj) == '[object Array]';
    }
}

instanceof

简单来说用来判断某个对象是否是由某个构造函数创建的.
严谨一点: 用来检查某个对象的构造函数的原型对象是否在当前对象的原型链上, 因为原型链可以任意由我们修改

示例代码如下:

function Person(){}
var person = new Person();
console.log(person instanceof Person); // true
Person.prototype = {};
console.log(person instanceof Person); // false


原文:https://www.jianshu.com/p/1016160e91fe

 

内容概要:本文是一篇关于使用RandLANet模型对SensatUrban数据集进行点云语义分割的实战教程,系统介绍了从环境搭建、数据准备、模型训练与测试到精度评估的完整流程。文章详细说明了在Ubuntu系统下配置TensorFlow 2.2、CUDA及cuDNN等深度学习环境的方法,并指导用户下载和预处理SensatUrban数据集。随后,逐步讲解RandLANet代码的获取与运行方式,包括训练、测试命令的执行与参数含义,以及如何监控训练过程中的关键指标。最后,教程涵盖测试结果分析、向官方平台提交结果、解读评估报告及可视化效果等内容,并针对常见问题提供解决方案。; 适合人群:具备一定深度学习基础,熟悉Python编程和深度学习框架,从事计算机视觉或三维点云相关研究的学生、研究人员及工程师;适合希望动手实践点云语义分割项目的初学者与进阶者。; 使用场景及目标:①掌握RandLANet网络结构及其在点云语义分割任务中的应用;②学会完整部署一个点云分割项目,包括数据处理、模型训练、测试与性能评估;③为参与相关竞赛或科研项目提供技术支撑。; 阅读建议:建议读者结合提供的代码链接和密码访问完整资料,在本地或云端环境中边操作边学习,重点关注数据格式要求与训练参数设置,遇到问题时参考“常见问题与解决技巧”部分及时排查。
内容概要:本文详细介绍了三相异步电机SVPWM-DTC(空间矢量脉宽调制-直接转矩控制)的Simulink仿真实现方法,结合DTC响应快与SVPWM谐波小的优点,构建高性能电机控制系统。文章系统阐述了控制原理,包括定子磁链观测、转矩与磁链误差滞环比较、扇区判断及电压矢量选择,并通过SVPWM技术生成固定频率PWM信号,提升系统稳态性能。同时提供了完整的Simulink建模流程,涵盖电机本体、磁链观测器、误差比较、矢量选择、SVPWM调制、逆变器驱动等模块的搭建与参数设置,给出了仿真调试要点与预期结果,如电流正弦性、转矩响应快、磁链轨迹趋圆等,并提出了模型优化与扩展方向,如改进观测器、自适应滞环、弱磁控制和转速闭环等。; 适合人群:电气工程、自动化及相关专业本科生、研究生,从事电机控制算法开发的工程师,具备一定MATLAB/Simulink和电机控制理论基础的技术人员。; 使用场景及目标:①掌握SVPWM-DTC控制策略的核心原理与实现方式;②在Simulink中独立完成三相异步电机高性能控制系统的建模与仿真;③通过仿真验证控制算法有效性,为实际工程应用提供设计依据。; 阅读建议:学习过程中应结合文中提供的电机参数和模块配置逐步搭建模型,重点关注磁链观测、矢量选择表和SVPWM调制的实现细节,仿真时注意滞环宽度与开关频率的调试,建议配合MATLAB官方工具箱文档进行参数校准与结果分析。
### JavaScript 继承的方法 #### 1. 原型链继承 原型链是 JavaScript 实现继承的一种基本模式。在这种模式下,子类通过其原型(prototype)链接到父类的实例上。这样,当访问子类的一个属性或方法时,如果该属性或方法不存在,则会沿着原型链向上查找直到找到为止。 然而,这种方式存在一些缺点,比如所有子类型的实例都会共享同一个超类型实例上的属性,这可能导致意外的状态共享问题[^1]。 ```javascript function SuperType() { this.property = true; } SuperType.prototype.getSuperValue = function () { return this.property; }; function SubType() {} // 设置SubType的原型为SuperType的新实例 SubType.prototype = new SuperType(); SubType.prototype.constructor = SubType; var instance = new SubType(); console.log(instance.getSuperValue()); // 输出: true ``` #### 2. 构造函数继承 (借用构造函数) 为了克服原型链的一些局限性,可以采用构造函数继承方式。这种方法是在子类构造器内部调用父类构造器,并传递当前 `this` 上下文给它。这样做可以让每个子类都有自己的独立副本而不是共享父类的数据成员。 但是这种做法也有不足之处——无法复用父类定义好的公共方法;每次创建新对象都要重新执行一次完整的初始化过程[^3]。 ```javascript function Parent(name) { this.name = name || 'parent'; } Parent.prototype.sayName = function(){ console.log(this.name); }; function Child(name){ Parent.call(this,name); // 调用了父级构造函数并传入参数 } let childInstance = new Child('child'); childInstance.sayName(); // 报错,因为Child没有sayName这个方法 ``` 注意上述代码中虽然实现了数据层面的继承,但并没有复制父类原型上的方法。 #### 3. 组合继承 组合继承综合了前两种方式的优点:既可以通过构造函数获得私有属性的支持,又可以从原型链那里得到可重用的功能。具体来说就是先利用构造函数完成对实例属性的操作,再让子类的原型指向一个父类实例来获取公有的行为。 不过这也带来了重复调用父类构造器的问题 —— 创建子类实例时以及设置子类原型的时候各发生了一次. ```javascript function Parent(name) { this.name = name || 'parent'; this.colors = ['red', 'blue']; } Parent.prototype.sayName = function(){ console.log(this.name); }; function Child(name, age){ Parent.call(this, name); // 第二次调用Parent() this.age = age; } Child.prototype = new Parent(); // 第一次调用Parent(), 并且这里丢失了constructor指针 Child.prototype.constructor = Child; const kid = new Child('kid', 5); console.log(kid instanceof Child); // true console.log(kid instanceof Parent); // true console.log(kid.sayName === Parent.prototype.sayName); //true ``` #### 4. 寄生组合式继承 为了避免组合继承中存在的效率低下问题,寄生组合式继承应运而生。此技术只调用一次父类构造器即可达到目的,而且不会影响原有功能。其实现原理在于借助临时性的中间件作为桥梁连接起两个类之间的关系,从而避免不必要的资源浪费。 这是目前被认为是最优解之一的做法[^2]。 ```javascript function inheritPrototype(subClass, superClass) { const prototype = Object.create(superClass.prototype); prototype.constructor = subClass; subClass.prototype = prototype; } inheritPrototype(Child, Parent); ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值