用 typescript 写了个函数,频频报错,请教如何解决

2021-02-03 12:07:59 +08:00
 november

typescript 初学者开始用 typescript 试着写点东西,却各种报错。遂来请教各位。

下面的函数作用是将两个对象合并,并返回合并结果。

function _merge<T extends U, U> (target: T, source: U) : T {
  for (let key in source) {
    if (Object.prototype.hasOwnProperty.call(source, key)) {
      const value = source[key]
      if (isObject(target[key]) && isObject(value)) {
        target[key] = _merge(target[key], value)
      } else if (isObject(value)) {
        target[key] = _merge({}, value)
      } else if (isArray(value)) {
        target[key] = _merge([], value)
      }
    }
  }
  return target
}

然后在target[key] = _merge({}, value)这段代码中的target[key]的错误是

Type 'U[Extract<keyof U, string>]' is not assignable to type 'T[Extract<keyof U, string>]'.
 Type 'U' is not assignable to type 'T'.
  'T' could be instantiated with an arbitrary type which could be unrelated to 'U'.ts(2322)

{} 又有错误

Argument of type '{}' is not assignable to parameter of type 'U[Extract<keyof U, string>]'.ts(2345)

最后在 target[key] = _merge([], value) 在的 [] 也有问题

Argument of type 'undefined[]' is not assignable to parameter of type 'never'.ts(2345)

请求如何解决上面的问题?

1773 次点击
所在节点    问与答
13 条回复
november
2021-02-03 13:02:49 +08:00
求个大佬给个解决思路。
BingoXuan
2021-02-03 13:06:47 +08:00
要不看一下 Object.assign 的签名
Justin13
2021-02-03 13:07:18 +08:00
november
2021-02-03 13:11:40 +08:00
@BingoXuan 请问怎么查看签名?
BingoXuan
2021-02-03 13:22:30 +08:00
@november
我在 webstorm 里面对应的签名来自 https://github.com/microsoft/TypeScript/blob/master/lib/lib.es2015.core.d.ts
签名是 assign<T, U>(target: T, source: U): T & U;
SxqSachin
2021-02-03 13:32:59 +08:00
同等大佬回答,我自己写这类方法的时候是最后返回的时候带一个 as T
sillydaddy
2021-02-03 13:53:55 +08:00
加一个 as any,就可以解决:
target[key] = _merge({} as any, value)
target[key] = _merge([] as any, value)

虽然取巧(因为没有明确类型),但因为显而易见,所以不会造成错误。否则需要好好啃一啃 TypeScript 的类型系统。
mxT52CRuqR6o5
2021-02-03 14:04:52 +08:00
写不出类型安全的很正常,你这种情况应该就是写不出来的,用各种 any 绕过去就行了
其他强类型语言基本都写不出这种通用的 merge 函数吧,java 如果用反射来实现保证不了类型安全吧
chenluo0429
2021-02-03 14:08:34 +08:00
你虽然用了 typescript,但是你的代码实际上一点都不 type 。_merge 的参数要求 target 的类型为 source 的子类,但是_merge({}, value)中{}怎么保证是 value 的子类?你写了不符合 type 的脏代码,要么 ts-ignore,要么就上 any
november
2021-02-03 14:19:24 +08:00
@BingoXuan
我用的 vscode,我想应该是一样的。不过我这里不是函数声明的时候报错,而是内部实现时,赋值报错。


@sillydaddy @mxT52CRuqR6o5
对象那行用 any 可以绕过去,但是数组那行还是报错说,不能赋值给 nerver 。
november
2021-02-03 14:35:48 +08:00
@chenluo0429

说得有点道理,那我改成了这样子:_merge<T, U> (target: T, source: U) : (T & U)

但是这样的话,每次访问 T[key] 的时候,除非进行断言 (<T&U>T)[key],否则都报错。
Type 'Extract<keyof U, string>' cannot be used to index type 'T'.ts(2536)

另外,(<T&U>target)[key] = _merge({}, value)这一行,对于(<T&U>target)[key]依然有错误。

Type '{} & U[Extract<keyof U, string>]' is not assignable to type '(T & U)[Extract<keyof U, string>]'.ts(2322)

请问这一行怎么弄?
mxT52CRuqR6o5
2021-02-04 13:31:28 +08:00
@november
因为 array 也是 object,你这条分支是走不到的,类型就成了 never
november
2021-02-04 15:43:36 +08:00
@mxT52CRuqR6o5
啊,是,你说得对。平时摸鱼太多了,一些基本知识都忽略了。。。。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/750875

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX