我不知道的Set

早上朋友发来一道面试题

印象中Set的机制是会把初始时候的可迭代对象调用next再一个个add进去,自然而然的想到了-0

javascript 复制代码
console.log(...new Set([null, null, undefined, undefined, NaN, NaN, -0, 0, +0]));// null undefined NaN 0

意料之外的结果,为什么是0?

Set如何判断重复?

先回到经典问题:Set是如何判断重复的?我们可以手写一个低级Set:

手写一个Set

kotlin 复制代码
class xSet {
    constructor(iterable = []) {
        this.items = [];
        if (iterable[Symbol.iterator]) {
            for (let item of iterable) {
                this.add(item);
            }
        }
    }

    add(value) {
        if (!this.has(value)) {
            this.items.push(value);
        }
        return this;
    }

    delete(value) {
        const index = this.items.findIndex(item => item === value);
        if (index !== -1) {
            this.items.splice(index, 1);
            return true;
        }
        return false;
    }

    has(value) {
        return this.items.some(item => item === value);
    }

    clear() {
        this.items = [];
    }

   size() {
        return this.items.length;
    }

    [Symbol.iterator]() {
        let index = 0;
        const items = this.items;
        return {
            next() {
                if (index < items.length) {
                    return { value: items[index++], done: false };
                } else {
                    return { done: true };
                }
            }
        };
    }

    keys() {
        return this[Symbol.iterator]();
    }

    values() {
        return this[Symbol.iterator]();
    }

    entries() {
        let index = 0;
        const items = this.items;
        return {
            next() {
                if (index < items.length) {
                    return { value: [items[index], items[index++]], done: false };
                } else {
                    return { done: true };
                }
            }
        };
    }

    forEach(callbackFn, thisArg) {
        for (let item of this.items) {
            callbackFn.call(thisArg, item, item, this);
        }
    }
}
去重核心

可以观察到new xSet的流程其实会调用add方法,add的时候会调用has,那么核心来了,has的判断条件是 item === value

javascript 复制代码
console.log(...new xSet([null, null, undefined, undefined, NaN, NaN, -0, 0, +0]));//null undefined NaN NaN -0
修改去重条件

与预期不符合,根据上文我们知道核心其实是has里面的判断条件,所以我们封装一个判断相等的函数,这个函数逻辑里面NaN 与 NaN算相等

javascript 复制代码
const sameValueZero = (x, y) => {
    return x === y || (typeof x === 'number' && typeof y === 'number' && isNaN(x) && isNaN(y));
}

has(value) {
        return this.items.some(item => sameValueZero(item , value));
    }

很好现在NaN已经被处理了,回到问题的核心那么-0为什么会被0呢

为什么输入的-0会变成0

猜想

1. 控制台显示问题

2. 数组转换的-0

3. Set转换的-0

验证

1. 控制台显示问题

这个问题最好解决我们打印-0, 出于严谨在node环境和浏览器环境都打印。

可以排除是控制台的显示问题。

2. 数组转换的-0

以下测试浏览器和node环境结果一致

打印一个数组:

arduino 复制代码
console.log([-0,0,+0]);// [ -0, 0, 0 ]

很好又排除掉一个猜想。

3. Set转换的-0

带着拨云见日以及可能前功尽弃的心情来到第三个测试:

arduino 复制代码
const set = new Set()
set.add(-0)
console.log(set);// Set(1) { 0 }
console.log(...set.keys()); // 0

随着一个个结果的打印心里的石头也是放下了。

新猜想与解决

这时候突然萌生出一个离谱的想法: 我看见是0那么一定是0吗,有没有可能还有什么导致显示问题?

好了现在自己把自己困住了,知识体系不完整常常陷入这种困境,思考几分钟后我突然想到,那么我用代码证明他是0不就行了!

判断0 和 -0 可太简单了,请出我们的Object.is

vbnet 复制代码
console.log(Object.is(-0,0)); // false
console.log(Object.is(...set.keys(),0)); // true

这一刻终于释怀了,问题的原因终于定位到了。

调研Set的机制

激动的打开es官网搜索Set

es官方定义的Set

new的时候调用了add,根据查看es其他方法的经验大胆猜想add的时候转换了-0

add方法

CanonicalizeKeyedCollectionKey的处理

结论

到这里终于解决了为什么-0会变成0了。由于add的时候先转换了一次-0导致最后变成0。

补充

前面手写Set里面定义了sameValueZero方法,其实es官方自己有一个sameValueZero方法

sameValueZero方法

# Number::sameValueZero方法

完善自定义Set

其实只需要自定义一个CanonicalizeKeyedCollectionKey方法

自定义 CanonicalizeKeyedCollectionKey方法

ini 复制代码
const CanonicalizeKeyedCollectionKey = (value) => value === -0 ? 0 : value

修改add方法

kotlin 复制代码
 add(value) {
        const newValue = CanonicalizeKeyedCollectionKey(value)
        if (!this.has(newValue)) {
            this.items.push(newValue);
        }
        return this;
    }

测试

javascript 复制代码
console.log(...new xSet([null, null, undefined, undefined, NaN, NaN, -0, 0, +0])); // null undefined NaN 0

完整的xSet

kotlin 复制代码
const sameValueZero = (x, y) => {
    return x === y || (typeof x === 'number' && typeof y === 'number' && isNaN(x) && isNaN(y));
}
const CanonicalizeKeyedCollectionKey = (value) => value === -0 ? 0 : value

export class xSet {
    constructor(iterable = []) {
        this.items = [];
        if (iterable[Symbol.iterator]) {
            for (let item of iterable) {
                this.add(item);
            }
        }
    }

    add(value) {
        const newValue = CanonicalizeKeyedCollectionKey(value)
        if (!this.has(newValue)) {
            this.items.push(newValue);
        }
        return this;
    }

    delete(value) {
        const index = this.items.findIndex(item => item === value);
        if (index !== -1) {
            this.items.splice(index, 1);
            return true;
        }
        return false;
    }

    has(value) {
        return this.items.some(item => sameValueZero(item , value));
    }

    clear() {
        this.items = [];
    }

   size() {
        return this.items.length;
    }

    [Symbol.iterator]() {
        let index = 0;
        const items = this.items;
        return {
            next() {
                if (index < items.length) {
                    return { value: items[index++], done: false };
                } else {
                    return { done: true };
                }
            }
        };
    }

    keys() {
        return this[Symbol.iterator]();
    }

    values() {
        return this[Symbol.iterator]();
    }

    entries() {
        let index = 0;
        const items = this.items;
        return {
            next() {
                if (index < items.length) {
                    return { value: [items[index], items[index++]], done: false };
                } else {
                    return { done: true };
                }
            }
        };
    }

    forEach(callbackFn, thisArg) {
        for (let item of this.items) {
            callbackFn.call(thisArg, item, item, this);
        }
    }
}
相关推荐
前端小巷子2 分钟前
当 WebSocket 遇见 Socket.io
前端·javascript·面试
小浣熊喜欢揍臭臭31 分钟前
react+antd 可拖拽模态框组件
前端·javascript·react
我的账号昵称1 小时前
基于 elements3 包装的 可展开 table 组件
javascript·vue.js·elementui
Dxy12393102162 小时前
Python单例模式详解:从原理到实战的完整指南
javascript·python·单例模式
weixin_584121437 小时前
vue3+ts+elementui-表格根据相同值合并
前端·javascript·elementui
拉不动的猪9 小时前
前端小白之 CSS弹性布局基础使用规范案例讲解
前端·javascript·css
伍哥的传说9 小时前
React强大且灵活hooks库——ahooks入门实践之开发调试类hook(dev)详解
前端·javascript·react.js·ecmascript·hooks·react-hooks·ahooks
F2E_Zhangmo11 小时前
基于cornerstone3D的dicom影像浏览器 第二章,初始化页面结构
前端·javascript·vue·cornerstone3d·cornerstonejs
DoraBigHead11 小时前
手写 `new`、`call`、`apply`、`bind` + V8 函数调用机制解密
前端·javascript·面试
JQLvopkk11 小时前
C#通过HslCommunication连接西门子PLC1200,并防止数据跳动的通用方法
前端·javascript·算法