首页 > 解决方案 > 打字稿、泛型和重载

问题描述

我正在努力在 Typescript 中实现一些我想重载的函数,这些函数也使用泛型。我对这个结果感到非常困惑:

*请注意,我已经删除了不会导致问题的代码,因此请忽略实际功能的无用性。

这有效:

export function persistContactData<T>(
  contactData: T,
  callback?: () => void
): void;
export function persistContactData<T>(
  { contactData }: { contactData: T },
  callback?: () => void
): void;
export function persistContactData<T>(
  { contactData }: { contactData: T },
  callback?: () => void
) {
  // implementation
}

这没有(只有第一个有错误):


// *** Compile error: "This overload signature is not compatible with its implementation"
//
export function getPersistedContactData<T>(
  contactId: string,
  callback: (result?: T) => void
): void;

// No more errors, and if I remove the above signature everything is fine.
export function getPersistedContactData<T>(
  { contactId }: { contactId: string },
  callback: (result?: T) => void
): void;
export function getPersistedContactData<T>(
  { contactId }: { contactId: string },
  callback: (result?: T) => void
) {
  // implementation
}

我对这里的问题感到困惑,但我猜测通用参数是如何使用的?但这似乎不应该有所作为。

(我正在使用 Typescript 3.9.3)

编辑:我更新了没有任何未知类型

标签: javascripttypescriptgenericsfunctional-programmingoverloading

解决方案


问题是泛型参数让编译器假设形状可能定义预期类型,因为实际实现将定义实际参数形状。

如果删除了所有泛型并使用实际类型代替,则可以看到这一点。

这将起作用(但实际上可能不是预期的):

export function persistContactData<T>(
  contactData: T,  // <---- What is T?  ┐( ̄~ ̄)┌  Could be the right shape?
  callback?: () => void
): void;
export function persistContactData<T>(
  { contactData }: { contactData: T },
  callback?: () => void
): void;
export function persistContactData<T>(
  { contactData }: { contactData: T },
  callback?: () => void
) {
  // implementation
}

这不会:

// Error
export function persistContactData(
  contactData: string,  // <---- string is not compatible with the implementation
  callback?: () => void
): void;
export function persistContactData(
  { contactData }: { contactData: string },
  callback?: () => void
): void;
export function persistContactData(
  { contactData }: { contactData: string },
  callback?: () => void
) {
  // implementation
}

推荐阅读