我在上图中简单模拟了一处疑似重污染工厂的用水与排污系统模型,通过类比可视化帮助我理解JavaScript对象属性的4个特性:
索取,代表工厂从大自然中索取洁净水源,水表记录了流过水流传感器的用水量。
写脏,代表工厂使用过的污水排入自然环境中,类似脏数据。
水表,只要水表正在运行中,相关部门就能了解该工厂的用水情况。
总控,断电后水表和排污开关皆停止运行。
排污控制,其开关与否直接影响写脏能否进行。
回到JavaScript中,假设有对象factories,表示全部重污染工厂的集合:
varfactories = {}
一家名为sb-1的污染工厂进入相关部门监控范围,于是在factories中加入新属性:
Object.defineProperty(factories,'sb-1', {writable:true,enumerable:true,configurable:true})
writable:true在JavaScript中表示可写性,图中表示sb-1工厂的排污开关是打开状态;
enumerable:true在JavaScript中表示可枚举性,图中表示sb-1工厂的水表正常运行中,等着相关部门来查水表;
configurable:true在JavaScript中表示可配置性,图中表示sb-1工厂的总控台是通电运行状态。
除了上面作为例子的1号污染工厂sb-1,还有2号、3号、4号等等都在相关部门监管范围内。相关部门开始查水表,只需遍历factories,如果检测到属性值为"污水"触发警报:
for(letfactoryinfactories){console.log(`即将查工厂“${factory}`”的水表)// 即将查工厂“sb-1”的水表check(factories[factory])// 嘟嘟嘟!污水!限期整改!}
查水表的前提条件是:工厂的水表必须是正在运行中。
sb-1工厂向环境中排放未过滤污水,为了绕过审查,将工厂水表关掉了:
Object.defineProperty(factories,'sb-1', {value:"污水"enumerable:false,})
这样一来,相关部门再次遍历factories中的所有工厂,不在审查工厂sb-1,厂长得意地笑了,小聪明也。
相关部门也不是吃素的,立即采取措施,审查组到污染工厂将排污阀门关闭,整改1个月:
Object.defineProperty(factories,'sb-1', {writable:false})
这下,sb-1工厂一个月不能再写脏了,厂长有小聪明吖,偷偷把排污阀门又打开了:
Object.defineProperty(factories,'sb-1', {writable:true})
这一次相关部门被惹怒,没手软,果断出手整治,污染工厂直接歇菜:
Object.defineProperty(factories,'sb-1', {enumerable:false,writable:false,configurable:false})
水表关闭;排污阀门关闭。总控台一旦被断电,重污染工厂关门大吉。
相关部门要批量关闭多个污染工厂,会这样做:
Object.defineProperties(factories, {'sb-1':{configurable: flase},'sb-2':{configurable: flase},'sb-3':{configurable: flase}})
另外,getter和setter也属于属性特性,它们是函数:
Object.defineProperty(factories,'sb-1', {get:function(){return'污水'},set:function(){},writable:false,configurable:false})
如果工厂将getter函数的算法设计为:
当水源被检测时返回(排出)"净水",当未被检测时返回(排出)"污水",emmmm???
检测对象属性特性可以使用Object.getOwnPropertyDescriptor(),第一个参数传入对象,第二个参数传入要查询的属性名:
Object.getOwnPropertyDescriptor(factories,"sb-1")
现实世界里,工厂正将污水肆无忌惮地排入河里;
虚拟世界里,喷子和键盘侠们正在网络里捍卫“正义”。
原创作者:帅华君
原文链接:http://www.shuaihuajun.com/article/javascript-object-property-attribute/