jquery JavaScript中的Defaultdict等价物

mrwjdhj3  于 2023-06-22  发布在  jQuery
关注(0)|答案(8)|浏览(154)

在python中,你可以有一个defaultdict(int),它将int存储为值。如果你试图对字典中不存在的键进行'get',你会得到零作为默认值。
你能在JavaScript/jquery中做同样的事情吗

qvsjd97n

qvsjd97n1#

您可以使用JavaScript Proxy构建一个

var defaultDict = new Proxy({}, {
  get: (target, name) => name in target ? target[name] : 0
})

这使您可以在访问属性时使用与普通对象相同的语法。

defaultDict.a = 1
console.log(defaultDict.a) // 1
console.log(defaultDict.b) // 0
defaultDict['c'] += 20
console.log(defaultDict.c) // 20

为了使其更清晰一些,可以将其 Package 在构造函数中,或者使用类语法。

class DefaultDict {
  constructor(defaultVal) {
    return new Proxy({}, {
      get: (target, name) => name in target ? target[name] : defaultVal
    })
  }
}

const counts = new DefaultDict(0)
console.log(counts.c) // 0

**EDIT:**以上实现仅适用于原语。它也应该通过接受默认值的构造函数来处理对象。下面是一个实现,它应该可以使用原语和构造函数。

class DefaultDict {
  constructor(defaultInit) {
    return new Proxy({}, {
      get: (target, name) => name in target ?
        target[name] :
        (target[name] = typeof defaultInit === 'function' ?
          new defaultInit().valueOf() :
          defaultInit)
    })
  }
}

const counts = new DefaultDict(Number)
counts.c++
console.log(counts.c) // 1

const lists = new DefaultDict(Array)
lists.men.push('bob')
lists.women.push('alice')
console.log(lists.men) // ['bob']
console.log(lists.women) // ['alice']
console.log(lists.nonbinary) // []
r6l8ljro

r6l8ljro2#

查看pycollections.js

var collections = require('pycollections');

var dd = new collections.DefaultDict(function(){return 0});
console.log(dd.get('missing'));  // 0

dd.setOneNewValue(987, function(currentValue) {
  return currentValue + 1;
});

console.log(dd.items()); // [[987, 1], ['missing', 0]]
bihw5rsg

bihw5rsg3#

我不认为有对等的,但你总是可以写自己的。JavaScript中字典的等价物是一个对象,因此可以这样编写它

function defaultDict() {
    this.get = function (key) {
        if (this.hasOwnProperty(key)) {
            return key;
        } else {
            return 0;
        }
    }
}

那就这样叫吧

var myDict = new defaultDict();
myDict[1] = 2;
myDict.get(1);
bvpmtnay

bvpmtnay4#

可以使用Proxy构造一个快速的脏黑客

function dict(factory, origin) {
    return new Proxy({ ...origin }, {
        get(dict, key) {
            // Ensure that "missed" keys are set into
            // The dictionary with default values
            if (!dict.hasOwnProperty(key)) {
                dict[key] = factory()
            }

            return dict[key]
        }
    })
}

下面的代码:

n = dict(Number, [[0, 1], [1, 2], [2, 4]])

// Zero is the default value mapped into 3
assert(n[3] == 0)

// The key must be present after calling factory
assert(Object.keys(n).length == 4)
iqxoj9l9

iqxoj9l95#

Proxies绝对使语法最像Python,有一个名为defaultdict2的库,它提供了一个非常清晰和彻底的基于代理的实现,支持nested/recursive defaultdicts,这是我真正重视的,也是我在这个线程中迄今为止的其他答案中所缺少的。
也就是说,我倾向于使用基于函数的方法来保持JS更“香草”/“原生”,比如这个概念验证:

class DefaultMap {
  constructor(defaultFn) {
    this.defaultFn = defaultFn;
    this.root = new Map();
  }
  
  put(...keys) {
    let map = this.root;
    
    for (const key of keys.slice(0, -1)) {
      map.has(key) || map.set(key, new Map());
      map = map.get(key);
    }

    const key = keys[keys.length-1];
    map.has(key) || map.set(key, this.defaultFn());
    return {
      set: setterFn => map.set(key, setterFn(map.get(key))),
      mutate: mutationFn => mutationFn(map.get(key)),
    };
  }
  
  get(...keys) {
    let map = this.root;

    for (const key of keys) {
      map = map?.get(key);
    }

    return map;
  }
}

// Try it:
const dm = new DefaultMap(() => []);
dm.put("foo").mutate(v => v.push(1, 2, 3));
dm.put("foo").mutate(v => v.push(4, 5));
console.log(dm.get("foo")); // [1, 2, 3, 4, 5]
dm.put("bar", "baz").mutate(v => v.push("a", "b"));
console.log(dm.get("bar", "baz")); // ["a", "b"]
dm.put("bar", "baz").set(v => 42);
console.log(dm.get("bar", "baz")); // 42
dm.put("bar", "baz").set(v => v + 1);
console.log(dm.get("bar", "baz")); // 43

DefaultMap的构造函数接受一个返回叶节点默认值的函数。结构的基本操作是putget,后者是不言自明的。put生成一个嵌套键链,并返回一对函数,让您在这些键的末尾改变或设置叶节点。访问.root可以得到底层的Map结构。
如果我忽略了任何错误或错过了有用的功能,请随时留下评论,我会把它扔进去。

mrphzbgm

mrphzbgm6#

受@Andy Carlson的答案启发,这里有一个以稍微更Python的方式工作的实现:

class DefaultDict {
    constructor(defaultVal) {
        return new Proxy(
            {},
            {
                get: (target, name) => {
                    if (name == '__dict__') {
                        return target;
                    } else if (name in target) {
                        return target[name];
                    } else {
                        target[name] = defaultVal;
                        return defaultVal;
                    }
                },
            }
        );
    }
}

基本上,它还允许您检索“目标”的所有gotten和set值,类似于collections.defaultdict在Python中的工作方式。这使我们能够执行以下操作:

const myDict = new DefaultDict(0);

myDict['a'] += 1;
myDict['b'] += 2;
myDict['c'] += 3;

myDict['whatever'];

console.log(myDict.__dict__);
// {'a': 1, 'b': 2, 'c': 3, 'whatever': 0}
v7pvogib

v7pvogib7#

为了补充安迪·卡尔森的回答
如果你默认dict一个数组,你会在结果对象中得到一个toJSON字段。你可以通过解构一个新的对象来摆脱它。

const dd = new DefaultDict(Array);
//...populate the dict
return {...dd};
pftdvrlh

pftdvrlh8#

原来的答案似乎不适用于嵌套的情况。我做了一些修改,使其工作:

class DefaultDict {
    constructor(defaultInit) {
      this.original = defaultInit;
      return new Proxy({}, {
        get: function (target, name) {
          if (name in target) {
            return target[name];
          } else {
            if (typeof defaultInit === "function") {
              target[name] = new defaultInit().valueOf();
            } else if (typeof defaultInit === "object") {
              if (typeof defaultInit.original !== "undefined") {
                target[name] = new DefaultDict(defaultInit.original);
              } else {
                target[name] = JSON.parse(JSON.stringify(defaultInit));
              }
            } else {
              target[name] = defaultInit;
            }
            return target[name];
          }
        }
      });
    }
  }

  var a = new DefaultDict(Array);
  a["banana"].push("ya");
  var b = new DefaultDict(new DefaultDict(Array));
  b["orange"]["apple"].push("yo");
  var c = new DefaultDict(Number);
  c["banana"] = 1;
  var d = new DefaultDict([2]);
  d["banana"].push(1);
  var e = new DefaultDict(new DefaultDict(2));
  e["orange"]["apple"] = 3;
  var f = new DefaultDict(1);
  f["banana"] = 2;

不同的是,如果defaultInit是一个对象,我们需要返回对象的深层副本,而不是原始副本。

相关问题