• JS系列2-怎么把一个对象当做数组使用


    怎么把一个对象当做数组使用?

    我们知道在JS中对象和数组的操作方式是不一样的,但是我们可以通过封装,给对象加一层包装器,让它可以和数组拥有同样的使用方式。我们主要借助Object.keys()Object.values()Object.entries()Proxy

    Object.keys

    看一下MDN上的解释:

    Object.keys() 方法会返回一个由一个给定对象的自身可枚举属性组成的数组,数组中属性名的排列顺序和正常循环遍历该对象时返回的顺序一致。

    也就是Object.keys可以获取对象的所有属性名,并生成一个数组。

    var obj = { a0b1c2 };
    console.log(Object.keys(obj));

    // console: ['a', 'b', 'c']
    • 1

    Object.values

    看一下MDN上的解释:

    Object.values()方法返回一个给定对象自身的所有可枚举属性值的数组,值的顺序与使用for...in循环的顺序相同 ( 区别在于 for-in 循环枚举原型链中的属性 )。

    Object.values()返回一个数组,元素是对象上找到的可枚举属性值。

    var obj = { foo'bar'baz42 };
    console.log(Object.values(obj));
    // ['bar', 42]
    • 1

    Object.entries

    看一下MDN上的解释:

    Object.entries()方法返回一个给定对象自身可枚举属性的键值对数组,其排列与使用 for...in 循环遍历该对象时返回的顺序一致(区别在于 for-in 循环还会枚举原型链中的属性)。

    Object.entries()返回一个数组,元素是由属性名和属性值组成的数组。

    const obj = { foo'bar'baz42 };
    console.log(Object.entries(obj));

    // [ ['foo', 'bar'], ['baz', 42] ]
    • 1

    Proxy

    Proxy是JS最新的对象代理方式,用于创建一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。

    使用Proxy可以封装对象的原始操作,在执行对象操作的时候,会经过Proxy的处理,这样我们就可以实现数组操作命令。

    基础 get 示例

    const handler = {
      getfunction(obj, prop{
        return prop in obj ? obj[prop] : 37;
      }
    }

    const p = new Proxy({}, handler);
    p.a = 1;
    p.b = undefined;

    console.log(p.a, p.b)
    console.log('c' in p, p.c)
    • 1

    以上示例的中,当对象中不存在属性名时,默认返回值为37

    无操作转发代理

    使用Proxy包装原生对象生成一个代理对象p,对代理对象的操作会转发到原生对象上。

    let target = {};
    let p = new Proxy(target, {});

    p.a = 37;   // 操作转发到目标

    console.log(target.a);    // 37. 操作已经被正确地转发
    • 1

    我们要实现一下几个函数:forEachmapfilterreduceslicefindfindKeyincludeskeyOflastKeyOf

    实现数组函数

    forEach

    数组中的forEach函数定义:arr.forEach(callback(currentValue [, index [, array]])[, thisArg])

    数组中的forEach需要传入一个函数,函数的第一个参数是当前操作的元素值,第二个参数是当前操作的元素索引,第三个参数是正在操作的对象。

    对于对象,我们将参数定为:currentValue、key、target。我们可以使用Object.keys来遍历对象。

    Object.keys(target).forEach(key => callback(target[key], key, target))
    • 1

    这里需要targetcallback参数,我们通过函数封装一下

    function forEach(target, callback{
      Object.keys(target).forEach(key => callback(target[key], key, target))
    }
    • 1

    这样我们就可以使用以下方式调用:

    const a = {a1b2c3}
    forEach(a, (v, k) => console.log(`${k}-${v}`))

    // a-1
    // b-2
    // c-3
    • 1

    通过Proxy封装:

    const handler = {
      getfunction(obj, prop{
        return forEach(obj)
      }
    }

    const p = new Proxy(a, handler)
    p.forEach((v, k) => console.log(`${k}-${v}`))
    • 1

    以上方式当然是不行的,我们主要看最后一句,其执行方式和数组的forEach完全相同,我们在调用Proxy封装的对象时,获取数据时,会调用get函数,第一个参数为原生对象,第二个参数为属性名-forEach,在这里就要修改我们的forEach函数了。 首先p.forEach的参数是一个函数,因此我们代理对象的返回值需要接收一个函数作为参数,因此修改如下:

    function forEach(target{
      return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target))
    }
    • 1

    因此完成代码为:

    function forEach(target{
      return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target))
    }

    const handler = {
      getfunction(obj, prop{
        return forEach(obj)
      }
    }

    const a = {a1b2c3}
    const p = new Proxy(a, handler)
    p.forEach((v, k) => console.log(`${k}-${v}`))
    // a-1
    // b-2
    // c-3
    • 1

    我们应该把以上代码封装为模块,方便对外使用:

    const toKeyedArray = (obj) => {
      const methods = {
        forEach(target) {
          return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target));
        }
      }

      const methodKeys = Object.keys(methods)

      const handler = {
        get(target, prop) {
          if (methodKeys.includes(prop)) {
            return methods[prop](target)
          }

          return Reflect.get(...arguments)
        }
      }

      return new Proxy(obj, handler)
    }

    const a = { a1b2c3}
    const p = toKeyedArray(a)
    p.forEach((v, k) => console.log(`${k}-${v}`))
    • 1

    以上是forEach的实现和封装,其他函数的实现方式类似。

    全部源码如下:

    const toKeyedArray = (obj) => {
      const methods = {
        forEach(target) {
          return (callback) => Object.keys(target).forEach(key => callback(target[key], key, target));
        },
        map(target) {
          return (callback) =>
            Object.keys(target).map(key => callback(target[key], key, target));
        },
        reduce(target) {
          return (callback, accumulator) =>
            Object.keys(target).reduce(
              (acc, key) => callback(acc, target[key], key, target),
              accumulator
            );
        },
        forEach(target) {
          return callback =>
            Object.keys(target).forEach(key => callback(target[key], key, target));
        },
        filter(target) {
          return callback =>
            Object.keys(target).reduce((acc, key) => {
              if (callback(target[key], key, target)) acc[key] = target[key];
              return acc;
            }, {});
        },
        slice(target) {
          return (start, end) => Object.values(target).slice(start, end);
        },
        find(target) {
          return callback => {
            return (Object.entries(target).find(([key, value]) =>
              callback(value, key, target)
            ) || [])[0];
          };
        },
        findKey(target) {
          return callback =>
            Object.keys(target).find(key => callback(target[key], key, target));
        },
        includes(target) {
          return val => Object.values(target).includes(val);
        },
        keyOf(target) {
          return value =>
            Object.keys(target).find(key => target[key] === value) || null;
        },
        lastKeyOf(target) {
          return value =>
            Object.keys(target)
              .reverse()
              .find(key => target[key] === value) || null;
        }
      }

      const methodKeys = Object.keys(methods)

      const handler = {
        get(target, prop) {
          if (methodKeys.includes(prop)) {
            return methods[prop](target)
          }

          const [keys, values] = [Object.keys(target), Object.values(target)];
          if (prop === 'length'return keys.length;
          if (prop === 'keys'return keys;
          if (prop === 'values'return values;
          if (prop === Symbol.iterator)
            return function* ({
              for (value of values) yield value;
              return;
            };

          return Reflect.get(...arguments)
        }
      }

      return new Proxy(obj, handler)
    }

    const x = toKeyedArray({ a'A'b'B' });

    x.a;          // 'A'
    x.keys;       // ['a', 'b']
    x.values;     // ['A', 'B']
    [...x];       // ['A', 'B']
    x.length;     // 2

    // Inserting values
    x.c = 'c';    // x = { a: 'A', b: 'B', c: 'c' }
    x.length;     // 3

    // Array methods
    x.forEach((v, i) => console.log(`${i}${v}`)); // LOGS: 'a: A', 'b: B', 'c: c'
    x.map((v, i) => i + v);                         // ['aA', 'bB, 'cc]
    x.filter((v, i) => v !== 'B');                  // { a: 'A', c: 'c' }
    x.reduce((a, v, i) => ({ ...a, [v]: i }), {});  // { A: 'a', B: 'b', c: 'c' }
    x.slice(02);                                  // ['A', 'B']
    x.slice(-1);                                    // ['c']
    x.find((v, i) => v === i);                      // 'c'
    x.findKey((v, i) => v === 'B');                 // 'b'
    x.includes('c');                                // true
    x.includes('d');                                // false
    x.keyOf('B');                                   // 'b'
    x.keyOf('a');                                   // null
    x.lastKeyOf('c');                               // 'c'
    • 1

    本文由 mdnice 多平台发布

  • 相关阅读:
    springboot基于vue的漫画动漫周边商城网站 前后端分离
    Design A Youtube
    【PythonIQ测试系统】学了一招测试一下女友的智商,这离谱的反射弧~麻了麻了(终极智商测试)
    【毕业设计】基于深度学习的人脸专注度检测计算系统 - opencv python cnn
    详解数据管理、数据治理、数据资产管理
    Everything——检索神兵
    Mybatis
    leetcode134.加油站 贪心法求解 (c++版本)
    Web权限&权限划分
    spring cloud 快速上手系列 -> 02-配置中心 Config -> 021-Config服务端
  • 原文地址:https://blog.csdn.net/wuqing942274053/article/details/126328361