前言

记得刚找工作那会,几种数据类型是必问题,当时的答案一般都是七种——字符串(String)、数字(Number)、布尔(Boolean)、数组(Array)、对象(Object)、空(Null)、未定义(Undefined),时至今日,某些网络教程上还是这样的分类:

其实,随着 ECMAScript 的发展和完善,在 ES6(2015) 和 ES11(2020) 中,又分别增加了 Symbol 和 BigInt 两种类型,所以,完整的分类应该是下面这样的:

今天,我们就来看看 Symbol 到底是什么类型,为何要引入这样一个类型。

背景

我们都应该有个清晰的认识:任何新技术或者新概念的出现,必然是为了解决某一痛点的。

想想吧,我们为了起一个漂亮的、符合语义规则的属性名而绞尽脑汁时的痛苦,还要承受属性名可能冲突的折磨,那是一段不堪回首的往事!

而 Symbol 的出现正是为了拯救我们的头发,让它们不至于牺牲在这些琐碎的小事上,它们每一根都是那么珍贵,它们的归宿应该在更具价值的地方!

概念

symbol 是一种基本数据类型。Symbol() 函数会返回 symbol 类型的值,该类型具有静态属性和静态方法。它的静态属性会暴露几个内建的成员对象;它的静态方法会暴露全局的 symbol 注册,且类似于内建对象类,但作为构造函数来说它并不完整,因为它不支持语法:"new Symbol()"。

语法

直接使用 Symbol() 创建新的 symbol 类型,并用一个可选的字符串作为其描述。

Symbol([description])
  • description (可选) 字符串类型。对symbol的描述,可用于调试但不是访问symbol本身。请注意,即使传入两个相同的字符串,得到的 symbol 也不相等。
const symbol1 = Symbol();
const symbol2 = Symbol(42);
const symbol3 = Symbol('foo');console.log(typeof symbol1);
// expected output: "symbol"console.log(symbol2 === 42);
// expected output: falseconsole.log(symbol3.toString());
// expected output: "Symbol(foo)"console.log(Symbol('foo') === Symbol('foo'));
// expected output: false

上面的代码创建了三个新的 symbol 类型。 注意,Symbol("foo") 不会强制将字符串 “foo” 转换成 symbol 类型。它每次都会创建一个新的 symbol 类型。

下面带有 new 运算符的语法将抛出 TypeError 运算符的语法将抛出错误:

var sym = new Symbol(); // TypeError

特性

正如歌词“每个人都有他的脾气”所说,Symbol 也有它自己的特性:

  1. 没有两个 Symbol 的值是相等的。就像“世上没有两片相同的叶子”一样,任何两个 Symbol 数据的值都不会相等。
  2. Symbol 数据值可以作为对象属性名。高手一出手,就知有没有。这一下子就奠定了 Symbol 的江湖地位。要知道,在之前,对象的属性名是字符串的专属权利,就连数字也会被同化为字符串,可现在居然被 Symbol 虎口夺食,字符串大概也只能黯然伤神了吧。

用涂

根据 Symbol 的特性,它有以下通途。

命名冲突

JavaScript 内置了一个 symbol ,那就是 ES6 中的 Symbol.iterator。拥有 Symbol.iterator 函数的对象被称为 可迭代对象 ,就是说你可以在对象上使用 for/of 循环。

const fibonacci = {[Symbol.iterator]: function* () {let a = 1;let b = 1;let temp;yield b;while (true) {temp = a;a = a + b;b = temp;yield b;}}
};// Prints every Fibonacci number less than 100
for (const x of fibonacci) {if (x >= 100) {break;}console.log(x);
}

为什么这里要用 Symbol.iterator 而不是字符串?假设不用 Symbol.iterator ,可迭代对象需要有一个字符串属性名 ‘iterator’,就像下面这个可迭代对象的类:

class MyClass {constructor (obj) {Object.assign(this, obj);}iterator() {const keys = Object.keys(this);let i = 0;return (function* () {if (i >= keys.length) {return;}yield keys[i++];})();}
}

MyClass 的实例是可迭代对象,可以遍历对象上面的属性。但是上面的类有个潜在的缺陷,假设有个恶意用户给 MyClass 构造函数传了一个带有 iterator 属性的对象:

const obj = new MyClass({ iterator: 'not a function' });

这样你在 obj 上使用 for/of 的话,JavaScript 会抛出 TypeError: obj is not iterable 异常。

可以看出,传入对象的 iterator 函数覆盖了类的 iterator 属性。

这有点类似原型污染的安全问题,无脑复制用户数据会对一些特殊属性,比如 proto 和 constructor 带来问题。

这里的核心在于,symbol 让对象的内部数据和用户数据井水不犯河水。

由于 sysmbol 无法在 JSON 里表示,因此不用担心给 Express API 传入带有不合适的 Symbol.iterator 属性的数据。另外,对于那种混合了内置函数和用户数据的对象,你可以用 symbol 来确保用户数据不会跟内置属性冲突。

私有属性

由于任何两个 symbol 都是不相等的,在 JavaScript 里可以很方便地用来模拟私有属性。symbol` 不会出现在 Object.keys() 的结果中,因此除非你明确地 export 一个 symbol,或者用 Object.getOwnPropertySymbols() 函数获取,否则其他代码无法访问这个属性。

function getObj() {const symbol = Symbol('test');const obj = {};obj[symbol] = 'test';return obj;
}const obj = getObj();Object.keys(obj); // []// 除非有这个 symbol 的引用,否则无法访问该属性
obj[Symbol('test')]; // undefined// 用 getOwnPropertySymbols() 依然可以拿到 symbol 的引用
const [symbol] = Object.getOwnPropertySymbols(obj);
obj[symbol]; // 'test'

还有一个原因是 symbol 不会出现在 JSON.stringify() 的结果里,确切地说是JSON.stringify()会忽略symbol属性名和属性值:

const symbol = Symbol('test');
const obj = { [symbol]: 'test', test: symbol };JSON.stringify(obj); // "{}"

总结

symbol 具有以下特性:

  • 每个 symbol 都是独一无二的。
  • symbol 可用作对象名称。

~

~
本文完,感谢阅读!

~

学习有趣的知识,结识有趣的朋友,塑造有趣的灵魂!

我是〖编程三昧〗的作者 隐逸王,我的公众号是『编程三昧』,欢迎关注,希望大家多多指教!

你来,怀揣期望,我有墨香相迎! 你归,无论得失,唯以余韵相赠!

知识与技能并重,内力和外功兼修,理论和实践两手都要抓、两手都要硬!

ES6 中的 Symbol 是什么?相关推荐

  1. ES6中的 Symbol

    目录 1.概述 2.symbol值的生成 3.作为属性名的Symbol值 4.symbol数据的总结注意 ES5中:5个基本数据类型 :number,boolean,string,null,undef ...

  2. 听说ES6中新增了能够与众不同的Symbol数据类型

    前言 小伙伴们大家好.不知道大家有没有遇到这样一种情况:在我们日常开发中,有时候可能会用到一些别人提供的对象,并且业务需要想要在这个对象的基础上进行一些扩展,添加一些属性或方法等.这个时候如果我们不了 ...

  3. es6中symbol详解

    ES6中引入了第6种原始类型:Symbol 创建Symbol let firstName = Symbol(); let person = {}; person[firstName] = '欧阳不乖' ...

  4. ES6中的class是如何实现的?(附Babel编译的ES5代码详解)

    序言 这篇文章主要讲解面试某大厂遇到的一个问题 - ES6中的class语法的实现? ECMAScript 6 实现了class,class是一个语法糖,使得js的编码更清晰.更人性化.风格更接近面向 ...

  5. 使对象具有ES6中Iterator接口的实现方法

    es6中只有具有iterator接口的数组或者类数组(arguments)都可以使用for of来循环,但是对于对象来说不行,可以利用迭代器中的原理来给对象生成一个迭代器,实现让对象可以使用for o ...

  6. 什么?ES6 中还有 Tail Calls!

    前言 先吐槽一件事,最近把原先的 TOP 域名更换到 CN 域名,并且用 Gatsby 重建个人站点,之前是用采用 HTTPS 部署的方式绕过阿里云的域名备案系统.更换 CN 域名后,这招不管用了,? ...

  7. ES6中的新特性:Iterables和iterators

    文章目录 简介 什么是iteration Iterable对象 普通对象不是可遍历的 自定义iterables 关闭iterators 总结 简介 为了方便集合数据的遍历,在ES6中引入了一个iter ...

  8. es6判断对象key是否存在,ES6中获取对象的key

    ES6中获取对象的key const json = {'a': 123, 'b': 321}; Object.keys(json).forEach(key => { window.console ...

  9. 关于es6中常见的一些方法----对象篇

    好了,话不多说,是骡子是马拉出来溜溜,这篇文章笔者就来简单介绍下关于对象的一些方法 1.Object.is() Object.is() 方法判断两个值是否是相同的值. 参数: value1: 第一个需 ...

最新文章

  1. 在vs2005中使用loki的方法
  2. C++改变基类成员在派生类中的访问属性
  3. 【极值问题】【CF1063B】 Labyrinth
  4. 顺序查找的基本原理及实现
  5. OCR算法:CNN+BLSTM+CTC架构(VS15)
  6. Java的重载与覆盖,傻傻分不清!
  7. 超简单利用xposed框架破解钉钉打卡
  8. ipvs,ipvsadm的安装及使用
  9. 不到三千买iPhone12 网友:抢到算我输
  10. 使用Python模拟男人在长椅上选择座位的过程
  11. c++引用另一个类的方法_利用CVE20191132:Windows内核中的另一个NULL指针取消引用...
  12. linux程序文本,Linux之文本处理
  13. MYSQL 查看表空间占用情况
  14. 解析TCP/UDP协议的通讯软件
  15. 腾讯与360继续争,受益的是谁?
  16. Google 阅读器键盘快捷键
  17. 学Python需要安装什么软件?Python软件工具大全
  18. 2016年8月18日 星期四 --出埃及记 Exodus 16:19
  19. 离散制造,重复制造和流程制造总结
  20. 游戏运营必须知道的知识(一),入行必备!!

热门文章

  1. MVC4发布到IIS,出现HTTP 错误 404.0 - Not Found的解决方法
  2. 活动子项父项的复杂CSS选择器[重复]
  3. 无法加载身份验证插件“ caching_sha2_password”
  4. 如何删除旧的和未使用的Docker映像
  5. java经典问题算法大全_10道java经典算法!每一题都能提升你的java能力!
  6. win11搜索位置在哪 Windows11搜索位置的设置方法
  7. win11网络配置文件类型怎么更改 Windows11更改网络配置文件类型的步骤教程
  8. Thread多线程用法示例
  9. groovy 使用java类_深入学习java中的Groovy 和 Scala 类
  10. ❤️《JUC并发编程从入门到高级》(建议收藏)❤️