首頁 > 軟體

详解JavaScript的Symbol型别、隐藏属性、全域性登录档

2022-05-13 21:11:49

Symbol型别的使用

在前文《JavaScript的8种资料型别》中,我们已经简单的介绍过了JavaScriptSymbol型别,下面对其使用方法和使用场景做一个简单的介绍。

Symbol简介

Symbol型别是JavaScript中的一种特殊的型别,特殊在所有的Symbol型别值都互不相同。我们可以使用“Symbol”来表示唯一的值,下面是建立Symbol物件的案例:

let id = Symbol();

这样我们就建立了一个Symbol型别的值,并把这个值储存在了变数id中。

Symbol型别的描述

我们在建立一个Symbol型别变数的时候,可以在引数中传入一些秒属性的字串,用于描述这个变数的用途资讯。
例如:

let id1 = Symbol('狂拽酷炫吊炸天的小明的id');
let id2 = Symbol('低调奢华有内涵的婷婷的id');

Symbol型别在任何时候都是不同的,即使他们拥有相同的描述资讯,描述只是一个标签,除此之外就没有别的用途了,例如:

let id1 = Symbol('id');
let id2 = Symbol('id');
console.log(id1==id2);//false

这个标签存在的意义,个人认为和Symbol不能直观的看到内部具体值的特性有关,通过新增一个描述资讯,让我们对变数的用途有更直观的了解。

Symbol不会隐式转字串

JavaScript中的大多数型别都可以直接转换成字串型别输出,所以我们不能直观的看到它的值到底是什么,例如我们可以直接用alert(123)把数位123转换成字串弹出。
但是Symbol型别比较特殊,它不能直接转换,例如:

let id = Symbol();
alert(id);//报错,不能把Symbol型别转为字串 

JavaScript中的Symbol型别不能转成字串是由于其内在的防治语言混乱的“语言保护”机制,因为字串和Symbol在本质上有着区别,不应该将其中一个转换成另一个。

试想一下,如果Symbol可以转为字串,那么它就变成了一个生成独一无二字串的函数,就不再具备独立资料型别的必要。

如果我们真的想知道Symbol变数的值,我们可以使用.toString()方法,如下所示:

let id = Symbol('this is identification');
console.log(id.toString());//Symbol(this is identification);

或者使用.description属性,获取描述资讯:

let id = Symbol('加油,奥利给');
console.log(id.description);//加油,奥利给」

Symbol类似作为物件的属性键

根据JavaScript的规范,只有两种型别的值可以作为物件的属性键:

  1. 字串
  2. Symbol

如果使用其他型别,则会隐式的转为字串型别。物件的键在前面的章节有详细的介绍,这里不再重复。

建立Symbol键

Symbol作为键值有两种方法:
例1:

let id = Symbol('id');
let user = {};
user[id] = 'id value';//新增Symbol键
console.log(user[id]);//id value

例2:

let id = Symbol('id');
let user = {
	[id]:'id value',//注意这里的方括号	
};
console.log(user[id]);

以上两个案例展示了在物件中插入Symbol型别作为键的用法,需要注意的是,在存取属性时需要使用obj[id]而不是obj.id,因为obj.id代表的是obj[‘id’]

如果我们使用Symbol作为物件的键会有什么效果呢?

for…in中被跳过

Symbol非常明显的一个特征是,如果物件中使用Symbol作为键,那么使用for…in语句是存取不到Symbol型别的属性的。

举个例子:

let id = Symbol('id');
let user = {
	name : 'xiaoming',
	[id] : 'id',
};
for (let key in user) console.log(user[key]);

执行以上程式码,得到以下结果:

> xiaoming

可以发现,[id]物件的值没有被列印出来,说明在物件属性列表中,使用for … in会自动忽略Symbol型别的键。

同样的,Object.keys(user)也会忽略所有的Symbol型别的键。

这样的特效能带来非常有用的效果,例如我们可以建立只能自己能用的属性。

虽然我们没有办法直接获取到Symbol键,但是Object.assign方法能够复制所有的属性:

let id = Symbol();
let obj = {
    [id] : '123'
}
let obj2 = Object.assign({},obj);
console.log(obj2[id]);

这并不影响Symbol的隐藏属性,因为复制后的物件仍然无法获取Symbol键。

隐藏自定义属性

由于Symbol既不能直接转为字串,我们没有办法直观的获得它的值,又不能通过for … in获得物件的Symbol属性,也就是说,如果没有Symbol变数本身,我们就没有办法获得物件内部的对应属性。

因此,通过Symbol型别的键值,我们可以隐藏属性,这些属性只能我们自己存取,其他人都看不到我们的属性。

举个例子:

我们在开发的过程中,需要和同事“张三”合作,而这个张三建立了一个非常好用的工具ToolTool是一个物件型别,我们想白嫖张三的Tool,并在此基础上新增一些自己的属性。

我们就可以通过新增Symbol型别的键:

let tool = {//张三写好了的Tool
    usage : "Can do anything",
}
let name = Symbol("My tool obj");
tool[name] = "This is my tool";
console.log(tool[name]);

以上范例展示了如何在别人写好的物件上新增自己的属性,那么为什么要使用Symbol型别而不是常规的字串呢?

原因如下:

  • 物件tool是别人写好的程式码,原则上我们不应该去修改别人的程式码,这样会造成风险;
  • 避免命名冲突,我们直接使用字串很有可能会和别人原有的属性键冲突,造成严重的后果;使用Symbol永远不会发生命名冲突,因为Symbol都是不同的;
  • 别人无法存取Symbol型别的键,相当于不会和别人的程式码冲突;

错误示范:
如果我们不使用Symbol型别,很可能出现以下情况:

let tool = {//张三写好了的Tool
    usage : "Can do anything",
}

tool.usage = "Boom Boom";
console.log(tool.usage);

以上程式码由于重复使用”usage”,从而重写了原属性,会造成物件原功能异常。

Symbol全域性登录档

所有的Symbol变数都是不同的,即使他们有用相同的标签(描述)。
有些时候,我们希望通过一个字串名称(标签),存取同一个Symbol物件,例如我们在程式码的不同地方存取相同的Symbol

JavaScript会维护一个全域性的Symbol登录档,我们可以通过向登录档中插入Symbol物件,并为物件起一个字串名称存取该物件。

向登录档插入或者读取Symbol物件需要使用Symbol.for(key)方法,如果登录档中有名为key的物件,就返回该物件,否则就插入新物件再返回。

举个例子:

let id1 = Symbol.for('id');//登录档内没有名为id的Symbol,建立并返回
let id2 = Symbol.for('id');//登录档内已有名为id的Symbol,直接返回
console.log(id1===id2);//true

我们通过Symbol.for(key)就能以全域性变数的方式使用Symbol物件,并使用一个字串标记物件的名字。

相反的,我们还可以使用Symbol.keyFor(Symbol)反向的从物件获取名称。

举个例子:

let id = Symbol.for('id');//登录档内没有名为id的Symbol,建立并返回
let name = Symbol.keyFor(id);
console.log(name);//id

Symbol.keyFor()函数只能用在全域性Symbol物件上(使用Symbol.for插入的物件),如果用在非全域性物件上,就会返回undefined

举个例子:

let id = Symbol('id');//区域性Symbol
let name = Symbol.keyFor(id);
console.log(name);//undefined

系统Symbol

JavaScript有许多系统Symbol,例如:

  • Symbol.hasInstance
  • Symbol.iterator
  • Symbol.toPrimitive

它们各有用途,我们在后面的会逐步介绍道这些独特的变数。

总结

  • Symbol物件的值是唯一的;Symbol可以新增一个标签,并通过标签在全域性登录档中查询物件的实体;
  • Symbol作为物件的键无法被for … in探测到;
  • 我们可以通过Symbol到全域性登录档存取全域性的Symbol物件;

但是,Symbol并不是完全隐藏的,我们可以通过Object.getOwnPropertySymbols(obj)获取物件所有的Symbol,或者通过Reflect.ownKeys(obj)获取物件所有的键。

到此这篇关于JavaScript的Symbol型别、隐藏属性、全域性登录档的文章就介绍到这了,更多相关js Symbol型别、隐藏属性、全域性登录档内容请搜寻it145.com以前的文章或继续浏览下面的相关文章希望大家以后多多支援it145.com!


IT145.com E-mail:sddin#qq.com