Symbol

2018-07-05  本文已影响0人  不可不信缘_b32e

一、概述

ES6之前JavaScript有六种数据类型,分别是Number、String 、Boolean、Undefined、Null和Object.在ES6中新引入了一种新的原始数据类型Symbol,它表示独一无二的值。

Symbol值通过Symbol函数生成。也就是说对象的属性名现在可以有两种类型:一种是字符串;另一种就是Symbol类型。只要属性值是Symbol类型,就是独一无二的,不会与其他属性名产生冲突。

var s1 = Symbol('foo');
var s2 = Symbol('bar');
s1   //  Symbol(foo)
s2  // Symbol(bar)

s1.toString()   //"Symbol(foo)" 
s2.toString()  // "Symbol(bar)"

上面代码中,s1和s2都是Symbol值。

(1)注意Symbol函数的只表示对当前Symbol值的描述,因此相同参数的Symbol函数返回值是不相等的。

// 没有参数的情况
let s1 = Symbol();
let s2 = Symbol();

s1 === s2 // false

// 有参数的情况
let s1 = Symbol('foo');
let s2 = Symbol('foo');

s1 === s2 // false

上面代码中,s1和s2都是Symbol函数的返回值,而且参数相同,但是它们是不相等的。

(2)Symbol 值不能与其他类型的值进行运算,会报错。

let sym = Symbol('My symbol');

"your symbol is " + sym
// TypeError: can't convert symbol to string

二、作为属性名的Symbol

let mySymbol = Symbol();

// 第一种写法
let a = {};
a[mySymbol] = 'Hello!';

// 第二种写法
let a = {
  [mySymbol]: 'Hello!'
};

// 第三种写法
let a = {};
Object.defineProperty(a, mySymbol, { value: 'Hello!' });

// 以上写法都得到同样结果
a[mySymbol] // "Hello!"

上面代码通过方括号和Object.defineProperty,将属性名指定为Symbol值

注意,Symbol 值作为对象属性名时,不能用点运算符。

三、属性名的遍历

Symbol作为属性名,但该属性不会出现在for···infor···of循环中,因此也不会被Object.key()Object.getOwnPropertyName()JSON.string()返回.但是她不是私有属性,有一个Objecct.getOwnPropertySymbols方法可以获取Symbol属性名。

Objecct.getOwnPropertySymbols方法返回一个数组。

const obj = {};
let a = Symbol('a');
let b = Symbol('b');

obj[a] = 'Hello';
obj[b] = 'World';

const objectSymbols = Object.getOwnPropertySymbols(obj);

objectSymbols
// [Symbol(a), Symbol(b)]

四、Symbol.for()、Symbol.keyfor()

(1)Symbol.for方法可以让我们重新使用同一个Symbol值。它接受一个字符串作为参数,搜索有没有该参数作为名称的Symbol值。有,就返回这个Symbol值;没有就新建一个并返回。
let s1 = Symbol.for('foo');
let s2 = Symbol.for('foo');

s1 === s2 // true

上面代码中,s1和s2都是 Symbol 值,但是它们都是同样参数的Symbol.for方法生成的,所以实际上是同一个值。

(2)Symbol.keyFor方法返回一个已登记的 Symbol 类型值的key
let s1 = Symbol.for("foo");
Symbol.keyFor(s1) // "foo"

let s2 = Symbol("foo");
Symbol.keyFor(s2) // undefined

上面代码中,变量s2属于未登记的 Symbol 值,所以返回undefined。

上一篇 下一篇

猜你喜欢

热点阅读