Typescript - 从基类中的静态方法实例化子类,使用 args 并引用其他静态方法

Posted

技术标签:

【中文标题】Typescript - 从基类中的静态方法实例化子类,使用 args 并引用其他静态方法【英文标题】:Typescript -- instantiate child from a static method in base class, with args and referencing other static method 【发布时间】:2021-05-03 22:22:00 【问题描述】:

我正在尝试从基类中的静态方法实例化子类。我想正确键入我的基类,而不是使用 any 作为我所有静态方法的返回类型。我尝试了解决方案here,但它不适用于引用其他静态方法或接受参数的静态方法。如何正确地从 typescript 中的基类继承,并且仍然引用其他方法并接受参数?

class BaseClass 
  id: string;

  [key: string]: unknown;

  static getName() 
    return this.name.toUpperCase()
  

  static async find<T extends BaseClass>(this: new (...args: any[]) => T, id: string)
  : Promise<T> 
    const tableName = this.getName();

    const result: GetResult = db.find(tableName, id);

    return new this(result);
  


class Child extends BaseClass 
  name: string;

  static findOne(id: string): Promise<Child> 
    return this.find(id);
  


Child.find('abcd');

这会导致两个不同的错误

    Property 'getName' does not exist on type 'new (...args: any[]) =&gt; T'.(在find 方法中) Type 'BaseModel' is missing the following properties from type 'Child': name.(在findOne方法的返回类型中)

【问题讨论】:

【参考方案1】:

在基类的find 方法中,您应该指定它期望子类实现静态getName 方法,如下所示:

static async find<T>(this:  new (arg: GetResult): T  & typeof BaseClass, id: string): Promise<T>

特别是 new (arg: GetResult): T 为您带来构造函数,typeof BaseClass 为您带来静态成员。

我嘲笑了一些缺失的部分,它会进行类型检查。

type GetResult = string;

const db = 
    find: (a: string, b: string) => "bar",


class BaseClass 
  id: string = "bzzzz";

  [key: string]: unknown;

  static getName() 
    return 'NAME'
  

  static async find<T>(this:  new (arg: GetResult): T  & typeof BaseClass, id: string)
  : Promise<T> 
    const tableName = this.getName();

    const result: GetResult = db.find(tableName, id);

    return new this(result);
  


class Child extends BaseClass 
  name: string = "Child";

  static findOne(id: string): Promise<Child> 
    return this.find(id);
  


Child.find('abcd');

【讨论】:

我试图简化实现以使问题清晰。我正在寻找的答案是如何让类型与我的实现一起工作,因为我的实现适用于普通 JS。 我需要能够调用this.getName,因为我的“现实生活”实现取决于子类的名称 我没有将 id 和 name 标记为可选,因为它们不是可选的。 @JeremyGottfried 好的,我明白了。我编辑了答案,希望我能正确理解你。 Steel 嘲笑了一些东西(因为我不知道从哪里获取 name、id、db 以及 GetResult 是什么,但现在它使用 child getName。现在看起来还可以吗? 谢谢,看来这行得通!我只是对此感到困惑:this: new (arg: GetResult): T; getName: () =&gt; string 。为什么我不能使用T extends BaseClass?如果 T 从 BaseClass 扩展,那不保证它有 getName 方法吗?

以上是关于Typescript - 从基类中的静态方法实例化子类,使用 args 并引用其他静态方法的主要内容,如果未能解决你的问题,请参考以下文章

如何从基类中的函数调用重载函数?

从基类c ++的向量中的派生类调用虚拟方法[重复]

派生类不从基类继承重载方法

从基类调用静态函数

Python中静态方法和类方法的区别

重写(override)与重载(overload)的区别