大家好,我是若川。持续组织了6个月源码共读活动,感兴趣的可以点此加我微信 ruochuan12 参与,每周大家一起学习200行左右的源码,共同进步。同时极力推荐订阅我写的《学习源码整体架构系列》 包含20余篇源码文章。历史面试系列
JavaScript即将推出两个新的数据类型:Record 和 Tuple ,这俩是啥呢?其实就是一个只读的 Object 和 Array,其实在其它语言中已经有类似的数据类型了,例如 Python 中也有 Tuple(元祖)这一类型,作用也是一个只读的数组(在Python里叫只读的列表),一起来了解一下,这个特性是一个第2阶段提案(即差不多稳了),想要提前体验的,文末也有 polyfill 的使用教程!
基础写法
// Records
const myRecord = #{name: '01',age: 23
}// Tuple
const myTuple = #['1', '2', '3']
其实就是在原先的对象和数组前面加了个 #
可读特性
Record和Tuple的语法跟对象和数组是一样的,所以?
const myRecord = #{name: '01'
}const myTuple = #['1', '2']myRecord['age'] = 23 // error
myTuple.push('3') // error
为啥报错了啊?开头有提到哦~因为这两个类型是 只读的 Object 和 Array
非唯一性
在平时的开发中,数组与数组、对象与对象 都不适合直接用 ===
进行比较判断,因为每个生成的对象在内存中的地址都不一样
const obj1 = { name: '01' }
const obj2 = { name: '01' }
const objIsSame = obj1 === obj2 // falseconst arr1 = [1]
const arr2 = [1]
const arrIsSame = arr1 === arr2 // false
要想真正比较两个数组或对象是否相等(即我们想要的内容都一样),需要遍历递归去一一对比,而现在呢?Record和Tuple能否解决这一问题呢?
const record1 = #{ name: '01' }
const record2 = #{ name: '01' }
const recordIsSame = record1 === record2 // trueconst tuple1 = #[1]
const tuple2 = #[1]
const tupleIsSame = tuple1 === tuple2 // true
可以看到,只要内部内容一致,即使是两个分别生成的Record或Tuple比较一下,也是相等的
普通对象和数组的转换
我可以用对象 Record
和 Tuple
将普通的对象和数组转换
const myRecord = Record({ name: '01', age: 23 }); // #{ name: '01', age: 23 }
const myTuple = Tuple([1, 2, 3, 4, 5]); // #[1, 2, 3, 4, 5]
支持扩展运算符
我们也可以对Record和Tuple使用扩展运算符
const myTuple = #[1, 2, 3];
const myRecord = #{ name: '01', age: 23 };const newRecord = #{ ...myRecord, money: 0 } // #{ name: '01', age: 23, money: 0 }
const newTuple = #[ ...myTuple, 4, 5]; // #[1, 2, 3, 4, 5]
JSON方法扩展
现在不是有 JSON.parse
和 JSON.stringfy
两个方法嘛,据说草案中还提到一个不错的想法,那就是给 JSON 对象新增一个 parseImmutable
方法,功能应该就是直接将一个 Record字符串或Tuple字符串 解析成对应的Record和Tuple对象
提前体验
如果你想现在体验该功能,可以装一下babel的插件
# babel基本的库
yarn add @babel/cli @babel/core @babel/preset-env -D# Record和Tuple Babel polyfill
yarn add @babel/plugin-proposal-record-and-tuple @bloomberg/record-tuple-polyfill -D
在目录下创建 .babelrc
,内容如下:
{"presets": ["@babel/preset-env"],"plugins": [["@babel/plugin-proposal-record-and-tuple",{"importPolyfill": true,"syntaxType": "hash"}]]
}
再创建一个 index.js
,内容如下:
const tuple1 = #[1,2,3]
const tuple2 = #[1,2,3]const record1 = #{ name: '01' }
const record2 = #{ name: '02' }console.log(tuple1 === tuple2, record1 === record2)
执行一下babel的命令编译一下
./node_modules/.bin/babel index.js --out-file compiled.js
输出得到的 compiled.js
文件内容如下:
"use strict";var _recordTuplePolyfill = require("@bloomberg/record-tuple-polyfill");var tuple1 = (0, _recordTuplePolyfill.Tuple)(1, 2, 3);
var tuple2 = (0, _recordTuplePolyfill.Tuple)(1, 2, 3);
var record1 = (0, _recordTuplePolyfill.Record)({name: '01'
});
var record2 = (0, _recordTuplePolyfill.Record)({name: '02'
});
console.log(tuple1 === tuple2, record1 === record2);
最后执行 compiled.js
即可获得结果
node compiled.js
# Result: true false
@babel/plugin-proposal-record-and-tuple
更多用法见 Babel 官方文档https://babeljs.io/docs/en/babel-plugin-proposal-record-and-tuple#docsNav
应用场景
了解了那么多的内容,印象最深刻的应该就是 只读 这个特性,那么基于这个特性,Record 和 Tuple 有哪些应用场景呢?
用于保护一些数据,比如函数的返回值、对象内部的静态属性...
既然具有只读的特性,即不可变对象,那应该也可以作为对象的 key 值吧?
················· 若川简介 ·················
你好,我是若川,毕业于江西高校。现在是一名前端开发“工程师”。写有《学习源码整体架构系列》20余篇,在知乎、掘金收获超百万阅读。
从2014年起,每年都会写一篇年度总结,已经写了7篇,点击查看年度总结。
同时,最近组织了源码共读活动,帮助3000+前端人学会看源码。公众号愿景:帮助5年内前端人走向前列。
识别上方二维码加我微信、拉你进源码共读群
今日话题
略。分享、收藏、点赞、在看我的文章就是对我最大的支持~