JS class语法糖的深入剖析


Posted in Javascript onJuly 07, 2022

引言

在很早以前,写过一篇文章 “类”设计模式和“原型”设计模式——“复制”和“委托”的差异 ,大致意思就是说:代码复用,也就是继承、重写,有两种思路:1. 面向对象的类继承;2. 基于 JavaScript 原型链的原型继承;前者的主要特点是:复制,通俗来说就是把变量、属性再复制一份,后者的主要特点是:委托,通过属性的查找来实现的。

后来呢,深入了解 JavaScript 高级程序设计中的继承,包括构造函数继承、原型继承、组合继承、寄生组合继承,都有各自的缺点,有兴趣的朋友,可以看我这篇文章。

还有,本瓜特别记住:维基对 JavaScript 起源的解释

JavaScript的语言设计主要受到了Self(一种基于原型的编程语言)和Scheme(一门函数式编程语言)的影响。在语法结构上它又与C语言有很多相似。

最后,我的小结呢就是:JavaScript 本身的设计就是“通过原型委托”来实现代码复用的,结果 ES6 搞出了个 class 作为语法糖,其本身还是基于原型链,但又是为了实现面向对象,面向对象是基于 class 类那种“复制”来实现代码复用。

类 和 原型,是两种不同的东西,JS class 将二者混在了一起,别不别扭?

后来也看到一些文章说在 JS 中使用 class 类会造成一些困扰,所以更加坚定要减少使用 class 。

而实际上,本篇题目是:JS class 并不只是简单的语法糖,所以,本篇并不是为了说它不好,而是要说它的好的!

来吧,展翅!

class 第一个好:私有变量

如果不用 class , 还有什么更优雅的方法实现以下子类的私有变量吗?

class Person {
  constructor(firstName, lastName) {
    this.firstName = firstName;
    this.lastName = lastName;
  } // Person.constructor
  get FullName () {
    return this.firstName + " " + this.lastName;
  }
} // Person
class Employee extends Person {
  #salary;
  constructor(firstName, lastName, salary) {
    super(firstName, lastName);
    this.salary = salary;
  }
  get salary() {
    return this.#salary;
  }
  set salary(salary) {
    this.#salary = salary;
    console.log("Salary changed for " + this.fullName + " : $" + this.salary);
  }
} // Employee

设想下,我们用原型链的思路模拟(对象):

const Person = {
  set givenName(givenName) {
    this._givenName = givenName;
  },
  set familyName(familyName) {
    this._familyName = familyName;
  },
  get fullName() {
    return `${this._givenName} ${this._familyName}`;
  }
};
const test = Person; // 这里假设用 对象 模拟 类
test.givenName = "Joe";
test.familyName = "Martinez";
console.log("test.fullName", test.fullName); // Joe Martinez
console.log("test.givenName", test.givenName); // undefined
console.log("test._givenName", test._givenName); // Joe

没有实现私有属性 _givenName

而 class 可以将值存为私有,使得对象外部不能修改:

class 第二个好:super 继承

class 可以通过 super 更优雅的实现继承、和重写,比如:

class Cash {
  constructor() {
    this.total = 0;
  }
  add(amount) {
    this.total += amount;
    if (this.total < 0) this.total = 0;
  }
} // Cash
class Nickles extends Cash {
  add(amount) {
    super.add(amount * 5);
  }
} // Nickles

如果是按照老样子,原型链,它可能是这样的:

const Cash = function() {
  this.total = 0;
}; // Cash
Cash.prototype = {
  add : function(amount) {
    this.total += amount;
    if (this.total < 0) this.total = 0;
  }
}; // Cash.prototype
const Nickles = function() {
  Object.assign(this, new Cash());
  this.add = function(amount) {
    Cash.add.apply(this, amount);
  };
} // Nickles

读起来有点乱,this 指来指去,还有在构造函数中手动做的 assign 操作,这会增加代码执行耗时。

综上两点,JS class 还是非常有使用它的价值的,不用逃避,把它用在合适的场景,肯定会发现其魅力~~

以上就是JS class语法糖的深入剖析的详细内容,更多关于JS class语法糖的资料请关注三水点靠木其它相关文章!

Javascript 相关文章推荐
jQuery实现公告文字左右滚动的实例代码
Oct 29 Javascript
用jquery等比例控制图片宽高的具体实现
Jan 28 Javascript
jquery操作HTML5 的data-*的用法实例分享
Aug 17 Javascript
jquery简单图片切换显示效果实现方法
Jan 14 Javascript
jquery使用EasyUI Tree异步加载JSON数据(生成树)
Feb 11 Javascript
Vue实现virtual-dom的原理简析
Jul 10 Javascript
详谈vue+webpack解决css引用图片打包后找不到资源文件的问题
Mar 06 Javascript
详解vue数组遍历方法forEach和map的原理解析和实际应用
Nov 15 Javascript
javascript实现小型区块链功能
Apr 03 Javascript
新手入门带你学习JavaScript引擎运行原理
Jun 24 Javascript
微信浏览器下拉黑边解决方案 wScroollFix
Jan 21 Javascript
jQuery实现简易聊天框
Feb 08 jQuery
MutationObserver在页面水印实现起到的作用详解
Jul 07 #Javascript
js作用域及作用域链工作引擎
Promise静态四兄弟实现示例详解
Jul 07 #Javascript
Three.js实现雪糕地球的使用示例详解
二维码条形码生成的JavaScript脚本库
Jul 07 #Javascript
JS实现简单的九宫格抽奖
JS实现九宫格拼图游戏
You might like
浅谈php函数serialize()与unserialize()的使用方法
2014/08/19 PHP
ThinkPHP多语言支持与多模板支持概述
2014/08/22 PHP
PHP扩展开发入门教程
2015/02/26 PHP
thinkPHP实现MemCache分布式缓存功能
2016/03/23 PHP
PHP水印类,支持添加图片、文字、填充颜色区域的实现
2017/02/04 PHP
PHP实现简易计算器功能
2020/08/28 PHP
漂亮的提示信息(带箭头)
2007/03/21 Javascript
js生成的验证码的实现与技术分析
2014/09/17 Javascript
jquery结婚电子请柬特效源码分享
2015/08/21 Javascript
解析JavaScript面向对象概念中的Object类型与作用域
2016/05/10 Javascript
jQuery的deferred对象使用详解
2016/09/25 Javascript
jQuery中get方法用法分析
2016/12/07 Javascript
浅谈JS中的反柯里化( uncurrying)
2017/08/17 Javascript
详解如何使用webpack在vue项目中写jsx语法
2017/11/08 Javascript
JavaScript实现全选取消效果
2017/12/14 Javascript
通过jquery toggleClass()属性制作文章段落更改背景颜色
2018/05/21 jQuery
微信小程序模拟cookie的实现
2018/06/20 Javascript
vue脚手架搭建项目的兼容性配置详解
2018/07/17 Javascript
简述vue状态管理模式之vuex
2018/08/29 Javascript
深入解析ES6中的promise
2018/11/08 Javascript
vue请求本地自己编写的json文件的方法
2019/04/25 Javascript
Vue可自定义tab组件用法实例
2019/10/24 Javascript
原生JS利用transform实现banner的无限滚动示例代码
2020/06/15 Javascript
关于vue 结合原生js 解决echarts resize问题
2020/07/26 Javascript
python文件和目录操作方法大全(含实例)
2014/03/12 Python
Python读写配置文件的方法
2015/06/03 Python
Python实现的彩票机选器实例
2015/06/17 Python
Python实现的爬虫功能代码
2017/06/24 Python
对变量赋值的理解--Pyton中让两个值互换的实现方法
2017/11/29 Python
使用pandas批量处理矢量化字符串的实例讲解
2018/07/10 Python
Python3中的bytes和str类型详解
2019/05/02 Python
使用python绘制温度变化雷达图
2019/10/18 Python
python取均匀不重复的随机数方式
2019/11/27 Python
Python获取指定网段正在使用的IP
2020/12/14 Python
Luxplus荷兰:以会员价购买美容产品等,独家优惠
2019/08/30 全球购物
高中生旷课检讨书
2014/10/08 职场文书