5

我正试图将我的头绕在流程上,但我很难让它与 ES6 一起工作Map

考虑这个简单的案例(现场演示):

// create a new map
const m = new Map();

m.set('value', 5);

console.log(m.get('value') * 5)

流量抛出:

console.log(m.get('value') * 5)
               ^ Cannot perform arithmetic operation because undefined [1] is not a number.
References:
[LIB] static/v0.72.0/flowlib/core.js:532:     get(key: K): V | void;
                                                               ^ [1]

我也试过:

const m:Map<string, number> = new Map();

m.set('value', 5);

console.log(m.get('value') * 5)

但我得到了同样的错误

我相信这是因为 flow 认为值也可以是数字以外的东西,所以我尝试用严格的 setter 和 getter 包装地图(现场演示):

type MyMapType = {
    set: (key: string, value: number) => MyMapType,
    get: (key: string) => number
};

function MyMap() : MyMapType {
    const map = new Map();

    return {
        set (key: string, value: number) {
          map.set(key, value);
          return this;
        },
        get (key: string) {
          return map.get(key);
        }
    }
}


const m = MyMap();

m.set('value', 5);

const n = m.get('value');

console.log(n * 2);

但后来我得到了:

get (key: string) {
^ Cannot return object literal because undefined [1] is incompatible 
with number [2] in the return value of property `get`.
References:
[LIB] static/v0.72.0/flowlib/core.js:532:     get(key: K): V | void;
                                                               ^ [1]
get: (key: string) => number                            ^ [2]

我如何告诉流程我只处理数字地图?

编辑:

Typescript 方法对我来说更有意义,它在set上而不是在get上抛出。

// TypeScript

const m:Map<string, number> = new Map();

m.set('value', 'no-number'); // << throws on set, not on get

console.log(m.get('value') * 2);

有没有办法让 Flow 的行为方式相同?

4

2 回答 2

8

Flow 试图告诉您的是,通过调用map.get(key).get(...)可能 ( V) 或可能不(void )从该映射中返回某些内容。如果在地图中找不到键,则调用.get(...)将返回undefined。为了解决这个问题,您需要处理返回未定义内容的情况。这里有几种方法:

试试

const m = new Map();

m.set('value', 5);

// Throw if a value is not found
const getOrThrow = (map, key) => {
  const val = map.get(key)
  if (val == null) {
    throw new Error("Uh-oh, key not found") 
  }
  return val
}

// Return a default value if the key is not found
const getOrDefault = (map, key, defaultValue) => {
  const val = map.get(key)
  return val == null ? defaultValue : val
}

console.log(getOrThrow(m, 'value') * 5)
console.log(getOrDefault(m, 'value', 1) * 5)

map.get(key)按原样键入的原因V | void是映射可能不包含该键的值。如果它的键没有值,那么您将引发运行时错误。Flow 开发人员决定他们宁愿强迫开发人员(你和我)在我们编写代码时考虑问题,然后在运行时找出问题。

于 2018-05-15T16:49:48.060 回答
1

随机而且很晚,但是当我没有看到它提到它时,我正在搜索并为我自己的用例提出了这个:

const specialIdMap = new Map<SpecialId, Set<SpecialId>>();
const set : Set<SpecialId> = specialIdMap.get(uniqueSpecialId) || new Set();

这节省了很多检查是否为空和/或其他的样板。当然,这仅在您也不依赖虚假值的情况下才有效。或者,您可以使用 new??运算符。

于 2019-12-27T18:21:48.007 回答