- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个“食物”对象,它可以是多种类型,具体取决于“类别” Prop 的值。该对象来自一个json,所以之前不可能知道类型。
我试图在 category 属性上使用 switch 语句,以便将 Food 对象转换为正确的类型
export type Category = 'fruit' | 'grain' | 'meat'
interface Food<IngredientCategory extends Category> {
name: string;
category: IngredientCategory
[key: string]: string;
}
interface Fruit extends Food<'fruit'> {
color: string;
}
interface Grain extends Food<'grain'>{
size: string;
}
interface Meat extends Food<'meat'> {
temperature: string
}
type FoodFromCategory<IngredientCategory extends Category> = IngredientCategory extends 'fruit' ? Fruit : IngredientCategory extends 'grain' ? Grain : Meat;
const castFood = <IngredientCategory extends Category>(category: IngredientCategory, food: Food<any>):
Food<IngredientCategory> | undefined => {
switch (category) {
case "fruit":
return food as Fruit
case "grain":
return food as Grain
case "meat":
return food as Meat
}
return undefined
};
这会在返回的行上产生错误:
TS2322: Type 'Fruit' is not assignable to type 'Food'. Types of property 'category' are incompatible. Type '"fruit"' is not assignable to type 'IngredientCategory'. '"fruit"' is assignable to the constraint of type 'IngredientCategory', but 'IngredientCategory' could be instantiated with a different subtype of constraint 'Category'.
为什么 switch 语句不缩小“IngredientCategory”通用参数的范围?还有其他方法可以做到这一点吗?
最佳答案
目前这在 TypeScript 中是不可能的。 control flow analysis这缩小了 category
的类型不会缩小类型参数 IngredientCategory
, 因此编译器无法推断出 Fruit
可分配给 FoodFromCategory<IngredientCategory>
即使category === "fruit"
.
在这里寻求一些解决方案的规范问题可能是 microsoft/TypeScript#33912 , 并且有一些特定的功能建议如果实现可能会允许这样做,例如 microsoft/TypeScript#33014 .
但是现在,编译器不能自己做这件事,如果你想编译它,你需要做一些像 type assertion 这样的事情。告诉编译器它可以处理(例如)Fruit
作为FoodFromCategory<IngredientCategory>
:
const castFood = <IngredientCategory extends Category>(
category: IngredientCategory, food: Food<any>
): FoodFromCategory<IngredientCategory> | undefined => {
switch (category) {
case "fruit":
return food as Fruit as FoodFromCategory<IngredientCategory>
case "grain":
return food as Grain as FoodFromCategory<IngredientCategory>
case "meat":
return food as Meat as FoodFromCategory<IngredientCategory>
}
return undefined
};
这有效并抑制了错误。但请注意,这是您将验证类型安全的责任从编译器上移开,并且您应该小心以确保正确执行此操作。
请注意,在运行时,switch 语句无论如何都没有做任何有用的事情。 JavaScript 不知道 Fruit
或 FoodFromCategory<IngredientCategory>
;所有这些都将作为 return food
发送给 JavaScript对于每个案例。如果你要去 return food
不管怎样category
是,你不妨写castFood
而是这样:
const castFood = <IngredientCategory extends Category>(
category: IngredientCategory, food: Food<any>
) => food as FoodFromCategory<IngredientCategory>;
这完全消除了 switch
陈述。它还消除了 undefined
,因为没有有效的方式来调用 castFood()
如果category
不在 IngredientCategory
中.
这就是所问问题的答案。退后一步,我有点担心你试图在这里做“ Actor ”; category
参数在运行时根本不使用,所以它只是帮助编译器确定什么类型 food
应该是。但在那种情况下,您可能最好使用 Fruit | Grain | Meat
作为discriminated union与共同category
属性,然后测试该属性,而不是将其作为单独的参数传递。
但是没有看到你打算如何或为什么调用 castFood()
我不会假设朝那个方向冒险(但也许它看起来像 this code )。相反,我只是重申,通过控制流缩小泛型类型参数目前是不可能的,您将需要使用类型断言或其他一些类型松散技术来编译它。
关于typescript - 如何使用 switch 语句来缩小使用条件类型的类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68474835/
是否有使用 switch 语句检查数字 0-9 的简单方法?我正在编写一个程序来检查某些字符和数字。就像检查“\0”、“F”或“f”一样,想知道是否还有一种方法可以以类似的方式检查 0-9。我知道我可
我有一些数据需要转换,为此我需要一个超过 50 个案例的切换条件,我需要 3 次相同的案例,但在第三次我需要 50 个案例和一些更多,我不想写两次相同的代码。也许有可能做这样的事情。 switch (
我遇到这种情况,我必须检查两个 GET 变量。在检查语句内的第一个 switch 语句后,必须在第一个 case 循环内的第二个 switch 语句中检查第二个变量。 我无法在这里发布确切的代码,但这
如何使用函数指针代替 switch 语句? 最佳答案 与 ars 发布的链接略有不同的方法:您可以将 switch 语句中的值用作函数指针数组中的数组索引。所以不要写 switch (i) {
我必须评估很多条件。就我而言,我必须做这样的事情: switch(id) { case 5: // switch some other cases here case
switch 按钮位于 switch 语句内,但仅在 switch 语句外部时才有效这是我的代码:
我试图在 switch 语句中有一个 case 跳转到不同的 switch 语句。 在实践中,我希望用户在文本框中键入“关闭页面”,并且在浏览器关闭页面之前,我希望询问用户是否确定。输入“yes”将关
(引用java)我试图确定哪个更好,编写更多代码并可能节省一些计算时间,或者编写更少代码但可能牺牲一些计算时间。这就是我好奇的地方这样做会更有效率吗: switch (availability) {
我正在尝试构建一个 Android 应用程序,该应用程序可以访问加速度计传感器,并在单击按钮时将加速度计值(由 <> 包围)输出到串行 USB。当我更新值并尝试在 onClick 命令中调用它时遇到问
如何使用 Dwoo 模板引擎实现 switch case 语法。 最佳答案 {if 3 == 5} never gonna happen {elseif 3 == 3} if you don'
我想知道一个大的 switch 语句和几个小的 switch 语句之间是否有性能差异。 包含数百个案例的大型 switch 语句。 switch(quest){ case 1:
用户在 2 个选择框中进行选择后,我尝试计算出报值(value)。 看起来 1 需要 2 个 switch 语句。这可能吗? (可能的值比下面的值多得多。为了清楚起见,我删除了它们) var wor
我在主 while 循环内有一个开关,它将运行我的游戏。我正在尝试打破我的开关,以便转到不同的案例。下面的例子更好地解释了这一点: int j = 0; While(1){ switch(j){ ca
我用 Java 创建了一个菜单,其中每个选项都有另一个菜单。我想知道是否可以从内部菜单退出回到主菜单。 编辑:添加了主菜单选项 System.out.println("Main Menu");
我有一个计算新分数的方法。下面的方法有效,但问题是代码本身看起来可以被显着清理。我只是不知道什么是最好的方法。我根据 filterString 和枚举 individualScoreState 分配
在 Lisp 中使用字符串切换语句。 (defun switch(value) (case value (("XY") (print "XY"))
我想做这样的事情: int i = 0; switch(difficulty) { case 1: i++; break; case 2: i--; break; defaul
在 Obj-c 中,我做了一个 switch 语句,我曾经使用 UIsplitviewcontroller 在我的 iPad 应用程序中四处移动现在我想快速地做同样的事情……我试了几个小时,现在我唯一
我想写一个结构如下的报告: \begin{document} \input[option=a]{class} \input[option=b]{class} \in
我正在认真阅读 ngSwitch 的 AngularJS API 引用。当我谈到那部分时的指令: place an expression on the on="..." attribute (or t
我是一名优秀的程序员,十分优秀!