利用 WeakMap 对 Vue 新建数组中的对象赋予 :key

时间: 2020-02-23阅读: 48标签: 对象

需求

在 Vue 中,对组件进行循环都需要加入key以便“就地复用”,可是在某些情况下,我们需要新建多个对象,而这些对象不是从后端获取到的,而是前端生成的,没有唯一值,且 Vue 目前版本只允许字符串,数字作为组件的 key。


方案

简单的组件

例如

<template> 
    <easy-component v-for="(item, index) in items" :key="index"/>
</template>
<script>
    export default{
        methods: {
            addSometing () {
                this.items.push({
                    // 一些属性
                    someProp
                })
            }
        }
    }
</script>

简单的组件,对 items 进行 CRUD 都是可以识别出来。不会影响界面的显示。

复杂的组件

但是对于一些复杂的组件 Vue 是识别不出来的,而且在删除时候会发生错乱。
所以需要这样写

<template> 
    <complex-component v-for="item in items :key="item.id"/>
</template>
<script>
    export default{
        methods: {
            addSometing () {
                this.items.push({
                    id: getUidFunction(),
                    // 一些属性
                    someProp
                })
            }
        }
    }
</script>

在创建时候添加唯一的 key —— id ,并且在上传的时候删除数组的 id

缺点

很难判断你所写的组件究竟是复杂还是简单,但在数组对象中添加唯一的 id 且必须在上传之前去除它,这终究不是一个好的解决方案。


更好的方法 WeakMap

思考

在ruby语言中,我们可以唯一确定这个对象,因为每个对象新建后都有一个唯一值确定该对象。但是 js 却没有这种语言特性。所以我们要从这方面入手考虑。

WeakMap的作用

WeakMap针对于普通的 Map 有两点特殊之处
1、WeakMap只接受对象作为键名( null 除外),不接受其他类型的值作为键名。
2、WeakMap的键名所指向的对象是弱引用,不计入垃圾回收机制。
重点在于 如果删除了WeakMap的键名所指向的对象,无需手动删除应用。
那么 思考后代码如下

// 唯一key
let uKey = 1
// 弱引用Map
const uidMap = new WeakMap()

function getUniqueKey (obj) {
    if (!uidMap.has(obj)) {
        uidMap.set(obj, uKey++)
    }
    return uidMap.get(obj)
}

// 为了简单直接使用插件
const uidPlugin = {
    install (Vue) {
        Vue.prototype.$uid = getUniqueKey
    }
}

if (typeof window !== 'undefined' && window.Vue) {
    window.Vue.use(uidPlugin)
}

export { uidPlugin }

在复杂的组件可以这样使用

<template> 
    <complex-component v-for="(item, index) in items :key="$uid(item)"/>
</template>

无需添加唯一的 id 以及删除 id ,即插即用且不影响垃圾回收。完美!


WeakMap 其他使用场景

1、标识 对象
2、缓存与对象相关的 属性
3、为对象添加监听器
具体可参考 Exploring ES6

原文:https://github.com/wsafight/personBlog/issues/1
站长推荐

1.阿里云: 本站目前使用的是阿里云主机,安全/可靠/稳定。点击领取2000元代金券、了解最新阿里云产品的各种优惠活动点击进入

2.腾讯云: 提供云服务器、云数据库、云存储、视频与CDN、域名等服务。腾讯云各类产品的最新活动,优惠券领取点击进入

3.广告联盟: 整理了目前主流的广告联盟平台,如果你有流量,可以作为参考选择适合你的平台点击进入

链接: http://www.fly63.com/article/detial/8344

在原生JavaScript中创建不可变对象

Javascript是一种灵活的语言,你可以重新定义任何东西,但是当项目变得复杂时,我们会发现可变数据结构的问题。随着JavaScript的最新版本的发布这种情况发生了改变。现在可以创建不可变的对象了。本文介绍如何用三种不同的方法来做。

浅谈JS包装对象

对象是 JavaScript 语言最主要的数据类型,三种原始类型的值——数值、字符串、布尔值——在一定条件下,也会自动转为对象,也就是原始类型的“包装对象(wrapper)。

vue中如何监听一个对象内部的变化?

方法1:对整个obj深层监听,默认第一次绑定的时候不会触发watch监听,值为true时可以在最初绑定的时候执行;方法2 :指定key;方法3:computed

Reflect对象

将Object 对象上的属于语言内部的方法放到 Reflect 对象上,从 Reflect 上获得语言内部的方法 ;修改某些 Object 方法的返回结果,让其变得更合理。让Object的操作都变成函数行为。Reflect 对象的方法与 Proxy 对象的方法一一对应。

javascript中document是什么?

javascript中document是window对象的属性,表示对Document对象的只读引用。Document对象是Window对象的一部分,可通过window.document属性对其进行访问。

javascript如何判断对象是否相等?

JavaScript作为一个基于对象(没有类的概念)的语言,从入门到精通到放弃一直会被对象这个问题围绕。下面我们就来看一下如何判断对象是否相等。

如何将JSON反序列化为JavaScript对象?

JSON(JavaScript Object Notation)用于与Web服务器或RESTFull API交换数据,从Web服务器接收的数据始终是字符串。为了使用这些数据,您需要使用JSON.parse()解析数据,它将返回一个JavaScript对象或对象数组。

File FileList 和 FileReader 对象详解

File 对象、FileList 对象与 FileReader 对象大家或许不太陌生,常见于文件上传下载操作处理(如处理图片上传预览,读取文件内容,监控文件上传进度等问题)

js内置对象

在js里,一切皆为或者皆可以被用作对象。可通过new一个对象或者直接以字面量形式创建变量,所有变量都有对象的性质。JS中常用的内置对象:Array对象、Date对象、正则表达式对象、string对象、Global对象

JS对象Object常用方法整理

hasOwnProperty():返回一个布尔值,指示对象自身属性中是否具有指定的属性(也就是,是否有指定的键)。isPrototypeOf():用于测试一个对象是否存在于另一个对象的原型链上。toString():返回一个表示该对象的字符串。

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!

文章投稿关于web前端网站点搜索站长推荐网站地图站长QQ:522607023

小程序专栏: 土味情话心理测试脑筋急转弯幽默笑话段子句子语录成语大全运营推广