当前位置: 首页 > news >正文

ES5 的构造函数和 ES6 的类有什么区别

文章目录

    • 语法不同
    • 方法定义方式不同
    • 继承方式不同
    • 类内部的this指向不同
    • 静态成员定义方式不同
    • 访问器属性
    • 类的类型检查

在JavaScript中,类和构造函数都被用来创建对象,接下来会从以下几点说说两者的区别:

语法不同

  • 构造函数使用函数来定义
  • 类使用class关键字来定义

ES6 的class可以看作是一个语法糖,这种写法只是让对象原型的写法更加清晰、更像面向对象编程的语法。

比如这是一个构造函数生成实例对象的例子,

function Point(x, y) {this.x = x;this.y = y;
}
Point.prototype.toString = function () {return '(' + this.x + ', ' + this.y + ')';
};
var p = new Point(1, 2);
console.log(p.toString()) // (1, 2)

上面的代码用类来改写是这样的,

class Point {constructor(x, y) {this.x = x;this.y = y;}toString() {return '(' + this.x + ', ' + this.y + ')';}
}
var p = new Point(1, 2);
console.log(p.toString()) // (1, 2)

所以类完全可以看作构造函数的另一种写法,

class Point {// ...
}
typeof Point // "function"
Point === Point.prototype.constructor // true

类的数据类型就是函数,类本身就指向构造函数。

方法定义方式不同

  • 构造函数的方法都是定义在构造函数的原型上,即原型上的方法被所有实例共享
  • 类的方法可以直接定义在类中,也可以定义在类的原型上,而且定义在类中的方法是不可枚举的。

构造函数的prototype属性,在 ES6 的“类”上面继续存在。事实上,类的所有方法都定义在类的prototype属性上面。

class Point {constructor() {// ...}toString() {// ...}toValue() {// ...}
}
// 等同于
Point.prototype = {constructor() {},toString() {},toValue() {},
};

以下代码表明类的内部所有定义的方法,都是不可枚举的(non-enumerable)

class Point {constructor(x, y) {// ...}toString() {// ...}
}
Object.keys(Point.prototype) // []
Object.getOwnPropertyNames(Point.prototype) // ["constructor","toString"]

而直接定义在原型上的都是可枚举的

class Point {constructor(x, y) {this.x = x;this.y = y;}toString() {return '(' + this.x + ', ' + this.y + ')';}
}
Point.prototype.toString1 = function() {// ...
};console.log(Object.keys(Point.prototype)) // [ 'toString1' ]
console.log(Object.getOwnPropertyNames(Point.prototype)) // [ 'constructor', 'toString', 'toString1' ]

继承方式不同

  • 构造函数通过将一个构造函数作为另一个构造函数的原型,实现原型链继承,可以实现单继承和多继承。
  • 类通过extends关键字来实现继承,支持单继承,子类继承父类的属性和方法,可以调用父类的构造函数。

构造函数的继承需要手动处理原型链,并确保构造函数和原型的正确指向,以下示例可看出

function Animal(name) {this.name = name;
}Animal.prototype.eat = function() {console.log(this.name + ' is eating.');
};function Dog(name, breed) {Animal.call(this, name); // 调用父类构造函数this.breed = breed;
}Dog.prototype = Object.create(Animal.prototype); // 设置子类的原型对象
Dog.prototype.constructor = Dog; // 重设构造函数指向自身Dog.prototype.bark = function() {console.log(this.name + ' is barking.');
};const dog = new Dog('Buddy', 'Golden Retriever');
dog.eat(); // 输出: Buddy is eating.
dog.bark(); // 输出: Buddy is barking.

类的继承示例

class Animal {constructor(name) {this.name = name;}eat() {console.log(`${this.name} is eating.`);}
}class Dog extends Animal {constructor(name, breed) {super(name); // 调用父类构造函数this.breed = breed;}bark() {console.log(`${this.name} is barking.`);}
}const dog = new Dog('Buddy', 'Golden Retriever');
dog.eat(); // Buddy is eating.
dog.bark(); // Buddy is barking.

需要注意的是,子类必须在constructor方法中调用super方法,否则新建实例时会报错。这是因为子类自己的this对象,必须先通过父类的构造函数完成塑造,得到与父类同样的实例属性和方法,然后再对其进行加工,加上子类自己的实例属性和方法。如果不调用super方法,子类就得不到this对象。
如果子类没有定义constructor方法,这个方法会被默认添加,代码如下。也就是说,不管有没有显式定义,任何一个子类都有constructor方法。

class Dog extends Animal {
}
// 等同于
class Dog extends Animal {constructor(...args) {super(...args);}
}

另一个需要注意的地方是,在子类的构造函数中,只有调用super之后,才可以使用this关键字,否则会报错。这是因为子类实例的构建,基于父类实例,只有super方法才能调用父类实例。

class Animal {constructor(name) {this.name = name;}
}
class Cat extends Animal {constructor(name, color) {this.color = color; // ReferenceErrorsuper(name);this.color = color; // 正确}
}

类内部的this指向不同

  • ES5中,类内部函数中的this指向调用函数的对象
  • ES6中,类内部方法中的this指向实例对象
    注意:如果静态方法中包含this关键字,这个this指的是类,而不是实例。
class Foo {static classMethod() {return 'hello';}}Foo.classMethod() // 'hello'
var foo = new Foo();
foo.classMethod()
// TypeError: foo.classMethod is not a function

静态成员定义方式不同

  • ES5中,静态成员需要通过给构造函数手动添加属性或方法来实现
function Person(name, age) {this.name = name;this.age = age;
}Person.staticMethod = function() {console.log("This is a static method.");
};
  • ES6中,可以使用static关键字定义静态属性或方法
class Person {constructor(name, age) {this.name = name;this.age = age;}static staticMethod() {console.log("This is a static method.");}
}

访问器属性

  • ES6中引入了getter和setter方法,可以使用get和set关键字定义访问器属性。
class Person {constructor(name, age) {this._name = name;this._age = age;}get name() {return this._name;}set name(value) {this._name = value;}
}var person = new Person("Tom", 20);
console.log(person.name); // 输出 "Tom"
person.name = "Jerry";
console.log(person.name); // 输出 "Jerry"

在上述例子中,通过getter和setter方法定义了name访问器属性,可以通过person.name进行属性的访问和修改。

类的类型检查

  • ES5中,可以使用instanceof运算符来判断对象的类型

instanceof 运算符通过检查对象的原型链来确定对象是否是某个构造函数的实例。它会逐级向上查找对象的 __proto__(或 prototype)属性,直到找到与指定构造函数的 prototype 属性相等的对象或到达原型链的末尾。如果找到了相等的对象,则返回 true,否则返回 false

function Person() {}
var person = new Person();
console.log(person instanceof Person); // 输出 true
  • ES6中,可以使用typeof运算符来判断类的类型
class Person {}
var person = new Person();
console.log(typeof person); // 输出 "object"
console.log(typeof Person); // 输出 "function"

在上述例子中,typeof person返回objecttypeof Person返回function,可以看出,在ES6中类的类型也是object,类本身的类型是function

http://www.lryc.cn/news/136145.html

相关文章:

  • AUTOSAR配置与实践(配置篇) 如何条件控制PDU外发
  • 2023年湖北中级工程师职称申报专业有哪些?甘建二告诉你
  • 记录:ubuntu20.04+ORB_SLAM2_with_pointcloud_map+ROS noetic
  • 文心问数Sugar Bot :大模型+BI,多轮会话自动生成可视化图表与数据结论
  • 21、WEB漏洞-文件上传之后端黑白名单绕过
  • windows的django项目部署到linux的docker上
  • 【力扣】70. 爬楼梯 <动态规划>
  • 数据结构(3)
  • 深入浅出Pytorch函数——torch.nn.init.xavier_uniform_
  • 优橙内推安徽专场——5G网络优化(中高级)工程师
  • 2023年计算机设计大赛国三 数据可视化 (源码可分享)
  • 工业生产全面感知!工业感知云来了
  • Lnton羚通关于Optimization在【PyTorch】中的基础知识
  • 冒泡排序算法
  • 无人机航管应答机 ping200XR
  • oracle归档日志满了导致启动不起来解决
  • 高等数学:线性代数-第二章
  • 星戈瑞分析FITC-PEG-Alkyne的荧光特性和光谱特性
  • VB.NET调用VB6 Activex EXE实现PowerBasic和FreeBasic的标准DLL调用
  • 深入了解Unity的Physics类:一份详细的技术指南(七)(下篇)
  • C++入门:引用是什么
  • 2023年人工智能与自动化控制国际学术会议(AIAC 2023)
  • 分布式核心知识以及常见微服务框架
  • Unity记录4.1-存储-根据关键字加载Tile
  • 数据结构—树表的查找
  • 微信小程序测试策略和注意事项?
  • VUE3封装EL-ELEMENT-PLUS input组件
  • RISC-V公测平台发布 · 在SG2042上配置Jupiter+Octave科学计算环境
  • 初识Sentinel
  • 【官方中文文档】Mybatis-Spring #注入映射器