Object对象的类似数组操作(map/reduce/forEach/filter/slice/find/findKey/includes/keyOf/lastKeyOf/)

通过Proxy对Object对象实现类似数组的操作,比如filter、map、slice、find等;
工具js代码:

const toKeyedArray = obj => {
  const methods = {
    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.defineProperties(target).find(([key, value]) => {
          return 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, receiver) {
      if (methodKeys.includes(prop)) return methods[prop](...arguments);
      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;
        };
      else return Reflect.get(...arguments);
    },
  }
  return new Proxy(obj, handler)
}

接下来进行测试:

const x = toKeyedArray({ a: 'A', b: 'B' })
console.log(x.a)//A
console.log(x.keys)//[ 'a', 'b' ]
console.log(x.values)//[ 'A', 'B' ]
console.log([...x])//[ 'A', 'B' ]
console.log(x.length)//2

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', b: 'B' }
x.reduce((a, v, i) => ({ ...a, [v]: i }), {});  // { A: 'a', B: 'b', c: 'c' }
x.slice(0, 2);                                  // ['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'

这就实现了Object的数组方法操作。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容