一个ts问题,很伤脑筋?

先说问题,renderType是一个联合类型,Shape类里面可能有一些方法,方法名必须是renderType里面的一个,然后Shape类里面需要有一个添加的方法,就是给Shape添加方法,添加的方法名也必须是renderType里面的一个,问题就出在这个添加方法customShape上,报错,说现在的类里面没有renderType的其他方法,不知道应该怎么改。

type renderType =
  | 'rectangle'
  | 'ellipse'
  | 'round'
  | 'triangle'
  | 'round-triangle'
  | 'round-rectangle'
  | 'bottom-round-rectangle'
  | 'cut-rectangle'
  | 'barrel'
  | 'rhomboid'

type returnSize = {
  width: number;
  height: number;
};

type ShapeType = {
  [key in renderType]?: (
    contentWidth: number,
    contentHeight: number
  ) => returnSize;
  // customShape: (shape: type.renderType, calcFunc: ( contentWidth: number, contentHeight: number ) => returnSize) => void
};

export class Shape implements ShapeType {
  rectangle(contentWidth: number, contentHeight: number) {
    return {
      width: contentWidth + 20,
      height: contentHeight + 10
    };
  }

  customShape(shape: renderType, calcFunc: Function): void {
    this[shape] = calcFunc;
  }
}

image.png

阅读 689
avatarAI BotBETA

你的代码中确实存在一些问题。在 TypeScript 里,当你尝试在 ShapeType 中使用 [key in renderType] 的时候,它期望 ShapeType 是一个索引签名类型。这意味着 ShapeType 中的每个属性都应该有一个特定的类型。在你的代码中,你为 [key in renderType] 提供了任意函数类型,这个函数接受两个数字参数并返回一个 returnSize 对象。

然而,你的 Shape 类中 customShape 方法试图给 ShapeType 添加一个方法,这个方法名应该是 renderType 中的某个值。然而,在 ShapeType 中并没有定义这个方法,所以 TypeScript 会报错,因为它不能保证 renderType 中的每个值都对应一个方法。

为了解决这个问题,你需要修改 ShapeType 的定义,使其包含一个额外的属性,这个属性是一个函数,该函数接受一个 renderType 类型的参数和一个函数参数,并返回 void。这样,你就可以在 Shape 类中添加方法了。

以下是修改后的代码:

type renderType =
  | 'rectangle'
  | 'ellipse'
  | 'round'
  | 'triangle'
  | 'round-triangle'
  | 'round-rectangle'
  | 'bottom-round-rectangle'
  | 'cut-rectangle'
  | 'barrel'
  | 'rhomboid'

type returnSize = {
  width: number;
  height: number;
};

type ShapeType = {
  [key in renderType]?: (
    contentWidth: number,
    contentHeight: number
  ) => returnSize;
} & {
  customShape: (shape: renderType, calcFunc: (contentWidth: number, contentHeight: number) => returnSize) => void;
};

export class Shape implements ShapeType {
  rectangle(contentWidth: number, contentHeight: number) {
    return {
      width: contentWidth + 20,
      height: contentHeight + 10
    };
  }

  customShape(shape: renderType, calcFunc: (contentWidth: number, contentHeight: number) => returnSize): void {
    this[shape] = calcFunc;
  }
}

这样,你的 Shape 类应该能正确地实现 ShapeType 接口,并且可以添加新的方法到 Shape 类中了。

2 个回答

在掘金的小册群里有大佬指点了一下,核心就是,显示的指定this,转成js的时候,这个this参数会被忽略,还是只有shape,calcFunc两个形参

customShape(this: ShapeType ,shape: renderType, calcFunc: Function): void {
    this[shape] = calcFunc;
  }

可以试试这个

 type TShapeCommonFunc = (contentWidth: number, contentHeight: number) => { width: number, height: number }

 interface IShape {
  rectangle?: TShapeCommonFunc,
  ellipse?: TShapeCommonFunc,
  round?: TShapeCommonFunc,
  triangle?: TShapeCommonFunc,
  roundTriangle?: TShapeCommonFunc,
  roundRectangle?: TShapeCommonFunc,
  bottomRoundRectangle?: TShapeCommonFunc,
  cutRectangle?: TShapeCommonFunc,
  barrel?: TShapeCommonFunc,
  rhomboid?: TShapeCommonFunc,
  addFunc: (newFunc: TShapeCommonFunc, funcName: Exclude<keyof IShape, 'addFunc'>)=>void
 }

 export class Shape implements IShape {
  ellipse: TShapeCommonFunc
  round: TShapeCommonFunc
  triangle:TShapeCommonFunc
  roundTriangle:TShapeCommonFunc
  roundRectangle:TShapeCommonFunc
  bottomRoundRectangle:TShapeCommonFunc
  cutRectangle:TShapeCommonFunc
  barrel:TShapeCommonFunc
  rhomboid:TShapeCommonFunc
  
  rectangle(contentWidth: number, contentHeight: number) {
    return {
      width: contentWidth + 20,
      height: contentHeight + 10
    };
  }

  addFunc(newFunc: TShapeCommonFunc, funcName: Exclude<keyof IShape, 'addFunc'>) {
    this[funcName] = newFunc
  }
 }

 // test --------------
 const b = new Shape();
 b.addFunc((contentWidth: number, contentHeight: number) => {
  console.log('--------------------')
  return {
    height: 1,
    width: 2,
  }
 },'ellipse')
 b.ellipse(2,3)
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题