ECMAScript6基础学习教程(五)对象

ES6在处理对象上,又添加了新方法。

1. 属性简洁写法

当属性名和属性值变量同名时,ES6允许在对象中只写属性名,不写属性。
关键点有两个:

  1. 属性值为变量
  2. 属性值变量名==属性名

例如:

var foo = "abc";
var obj = {
  name: 'nicole',
  foo
}; //相当于是 var obj = {foo: foo}

如果属性值是函数,可以省略关键字function,如下:

var obj = {
  say(name) {
    return "hello, " + name;
  }
}
//等价于
var obj = {
  say: function (name) {
    return "hello, " + name;
  }
}

2. 属性名表达式

ES5中,使用字面量方式定义对象时,属性名不能是变量。ES6中可以支持变量/表达式格式的属性名,格式为[propertyName]

var key = "property name";
var a = "hello";
var b = "world";
var obj = {
  [key]: 1,
  name: "nicole",
  [a+" "+b]: "good"
};
obj["property name"]; //1
obj["hello world"]; //"good"

注意:“属性简洁写法”和“属性名表达式”不能同时使用。

3. Object.is()

Object.is(val1, val2)用于判断两个值是否严格相等,用===判断。

注意:我们知道NaN===NaN会返回false,但是Object.is(NaN, NaN),返回true

4. Object.assign()

Object.assign(target, object1 [,objectN])用于对象拷贝,将对象object1...复制到对象target中。

Object.assign()是浅拷贝,类似jQuery.extend( false, target, object1 [, objectN ] )

var obj1 = {a: {b:1}, x: 1};
var obj2 = {x: 3, y: 4};
var target = {};
Object.assign(target, obj1, obj2);
// 改变复制源对象的值
obj1.a.b = 11;
// 复制对象同步改变,说明复制的仅仅是源对象引用地址
console.log(target); // { a: { b: 11 }, x: 3, y: 4 }

5. WeakMap和WeakSet

MapSet是对对象的增强类型,Map偏向对象增强,Set偏向数组增强。
MapObject最大的区别,就是Map的Key可以不是简单类型,比如,对象作为Key。并且,MapSet内置很多方法来管理和操作数据集合,比如:has, add, delete, clear 等。

WeakSet结构与Set类似,也是不重复的值的集合,特点如下:

  • 成员只能是对象
  • WeakSet中的对象都是弱引用,也就是说,如果其他对象都不再引用该对象,那么垃圾回收机制会自动回收该对象所占用的内存,不考虑该对象还存在于WeakSet之中。

同样的,WeakMap结构与Map结构基本类似,特点如下:

  • 只接受对象作为键名(null除外),
  • WeakMap中的对象都是弱引用

在计算机程序设计中,弱引用与强引用相对,是指不能确保其引用的对象不会被垃圾回收器回收的引用。 一个对象若只被弱引用所引用,则被认为是不可访问(或弱可访问)的,并因此可能在任何时刻被回收。

举个例子,如下两个对象elt1和elt2, 哪怕已经销毁了,由于array强引用了它们,否则垃圾回收机制不会释放elt1和elt2的内存。除非手动去掉引用:array[0]=null; array[1]=null;

const elt1 = document.getElementById("id1");
const elt2 = document.getElementById("id2");

const array = [elt1, elt2]

下面看看两种类型的使用场景,就更能了解了。

WeakSet

  1. 定义类
const requests = new WeakSet();
class ApiRequest {
  constructor() {
    requests.add(this);
  }
  makeRequest() {
    if(!request.has(this)) throw new Error("Invalid access");
    // do work
  }
}
  1. 创建DOM(通过加入对应集合,给这个节点打上“禁用”标签,只要 WeakSet 中任何元素从 DOM 树中被删除,垃圾回收程序就可以忽略其存在,而立即释放其内存)
const disabledElements = new WeakSet(); 
const loginButton = document.querySelector('#login'); 

WeakMap

  1. 注册监听事件的listener对象(由于监听函数是放在 WeakMap 里面,则一旦dom对象ele1,ele2消失,与它绑定的监听函数handler1和handler2 也会自动消失)
// 代码1
ele1.addEventListener('click', handler1, false);
ele2.addEventListener('click', handler2, false);
 
// 代码2
const listener = new WeakMap();
 
listener.set(ele1, handler1);
listener.set(ele2, handler2);
 
ele1.addEventListener('click', listener.get(ele1), false);
ele2.addEventListener('click', listener.get(ele2), false);
  1. 部署私有属性(Countdown类的两个内部属性_counter_action,是实例的弱引用,所以如果删除实例,它们也就随之消失,不会造成内存泄漏。)
let _counter = new WeakMap();
let _action = new WeakMap();

class Countdown {
  constructor(counter, action) {
    _counter.set(this, counter);
    _action.set(this, action);
  }
  dec() {
    let counter = _counter.get(this);
    if (counter < 1) return;
    counter--;
    _counter.set(this, counter);
    if (counter === 0) {
      _action.get(this)();
    }
  }
}

let c = new Countdown(2, () => console.log('DONE'));

c.dec()
c.dec()
  1. 数据缓存(当我们需要在不修改原有对象的情况下储存某些属性等,而又不想管理这些数据时,可以使用WeakMap)
const cache = new WeakMap();

function countOwnKeys(obj) {
  if (cache.has(obj)) {
     return cache.get(obj)
  }
  else {
      const count = Object.keys(obj).length
      cache.set(obj, count)
      return count
  }
}

小结

最常用的是“属性简洁写法”和“属性名表达式”。
做对象拷贝时,除了Object.assign(),更简便的方法是使用“扩展运算符”(参考上一节文章ECMAScript6基础学习教程(四)函数 - 扩展运算符)。

例子如下:

var obj1 = {a: {b:1}, x: 1};
var obj2 = {x: 3, y: 4};
var target = {...obj1, ...obj2};
// 改变复制源对象的值
obj1.a.b = 11;
// 复制对象同步改变
console.log(target); // { a: { b: 11 }, x: 3, y: 4 }

下一节:ECMAScript6基础学习教程(六)类

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • 1.属性的简洁表示法 允许直接写入变量和函数 上面代码表明,ES6 允许在对象之中,直接写变量。这时,属性名为变量...
    雨飞飞雨阅读 1,128评论 0 3
  • 属性的简洁表示法 ES6允许直接写入变量和函数,作为对象的属性和方法。这样的书写更加简洁。 上面代码表明,ES6允...
    呼呼哥阅读 2,903评论 0 2
  • 三,字符串扩展 3.1 Unicode表示法 ES6 做出了改进,只要将码点放入大括号,就能正确解读该字符。有了这...
    eastbaby阅读 1,510评论 0 8
  • 登录注册分为第三方登录(QQ、微信),立即注册、免密登录、忘记密码,整个模块细节非常多。 登录注册 登录注册 想要...
    葱花饼阅读 533评论 0 2
  • 第一板斧:确定你的职业状态 1. 兴趣高+能力低:业余爱好 2. 兴趣高+能力高:发展方向 3. 兴趣...
    加油吧蜗牛阅读 246评论 0 1