- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
让我们以example为例来自 TS 3.3 文档:
type Fruit = "apple" | "orange";
type Color = "red" | "orange";
type FruitEater = (fruit: Fruit) => number;
type ColorConsumer = (color: Color) => string;
type FnUnion = FruitEater | ColorConsumer
我想,以下函数可以分配给FnUnion
:
declare function foo(v: Fruit & Color): number | string
,正如有人告诉我的,函数在参数上是逆变的,在返回类型上是协变的。但请考虑以下内容:
declare let fn: FnUnion
declare function foo(v: Fruit & Color): number | string
declare function foo2(v: Fruit | Color): number
declare function foo3(v: Fruit | Color): string
fn = foo // error, why?
fn = foo2 // works
fn = foo3 // works
我对上述情况下联合函数的一般可分配性规则感兴趣。 foo2
和 foo3
可分配给 fn
。但为什么 foo
不可赋值呢?
我肯定会错过一些 TS 知识,感谢您的解答。谢谢! PS:这是一个sample .
最佳答案
需要明确的是,&
是交集,|
是并集。这意味着:
Fruit & Color //=> "orange"
Fruit | Color //=> "apple" | "orange" | "red"
这意味着:
type FnUnion = FruitEater | ColorConsumer
是接受所有水果的函数,或者是接受所有颜色的函数。
以下是该类型的有效用法:
const colorFn: FnUnion = (color: Color) => 'a color'
colorFn('red')
colorFn('orange')
const fruitFn: FnUnion = (fruit: Fruit) => 123
fruitFn('apple')
fruitFn('orange')
注意:必须是其中之一。它必须接受所有水果或它必须接受所有颜色。
但是,Fruit & Color
类型只能是'orange'
。因此,您将一个只能接受 'orange'
的函数分配给一个允许接受 'orange'|'red'
或'橙色'|'苹果'
。这意味着 'red'
或 'apple'
永远不能作为参数传递,其中 FnUnion
表示应允许其中之一.
因此 typescript 正确地提示类型不兼容。
此外,Typescript 知道联合体类型,还知道控制流中某些点的联合体成员的值。
当您像这样转换到 FnUnion
时:
const colorFn = ((color: Color) => 'a color') as FnUnion
您告诉 TS 它是联合体的哪个成员是不可知的,因为它不能再根据参数类型进行区分。
因此,似乎唯一安全的推论是这两种类型共有的一个参数。我意识到,这正是它不允许您直接分配的函数类型。
但我认为关键的区别是知道你的函数是工会的哪个成员。
// member known: ColorConsumer
const colorFn: FnUnion = (color: Color) => 'a color'
// member known: FruitEater
const fruitFn: FnUnion = (fruit: Fruit) => 123
// member unknown due to cast to less specific type.
// only common arguments are allowed,
// and only return types for those common arguments
const unknownFn: FnUnion = ((some: 'orange') => 'unknown') as FnUnion
// type is concretely known, but is not a match to any member.
const invalidFn: FnUnion = (some: 'orange') => 'invalid'
// ^ type error
FnUnion
表示它是其中之一,如果它不知道是哪一个,那么它是两者的交集。您想说的类型是已知的,它不是成员,而是两者的交集。这就是不兼容的地方。
但是正如您发现的那样,使用 as FnUnion
您可以使 typescript 忘记它对类型转换的了解。
所有这些意味着推断的某物类型并不总是它可以分配的类型。
关于typescript - 联合函数类型的可赋值性规则,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60440070/
你能解释一下这个作业是如何完成的吗, var fe, f = document.forms[0], h; 哪个等于哪个。 最佳答案 以上等同于 var fe; var f = document.for
据我测试,这两种方法都有效,但我不知道哪一种最好,也不知道它们之间的区别,这就是我想知道的。 以下是两种方法: window.location = 'http://www.google.com'; w
我正在处理用字符串填充的 numpy 数组。我的目标是分配给第一个数组 a 的切片,值包含在较小尺寸的第二个数组 b 中。 我想到的实现如下: import numpy as np a = np.em
在我使用过的其他语言(如 Erlang 和 Python)中,如果我正在拆分字符串并且不关心其中一个字段,我可以使用下划线占位符。我在 Perl 中试过这个: (_,$id) = split('
我认为这似乎很简单,但我对调用、应用、绑定(bind)感到困惑。等等 我有一个事件监听器 red.addEventListener("click", function() { j = 0;
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 使用有什么区别: iFile =
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 使用有什么区别: iFile =
几周前我们开始写一篇关于 Haskell 的论文,刚刚接到我们的第一个任务。我知道 SO 不喜欢家庭作业问题,所以我不会问怎么做。相反,如果有人能将我推向正确的方向,我将不胜感激。鉴于它可能不是一个特
我正在尝试为我的函数的变量根分配一个值,但似乎不起作用。我不明白这个问题。 hw7.c:155:7:警告:赋值使指针来自整数而不进行强制转换[默认启用] root = 负载(&fp, 大小); 此代码
我昨天花了大约 5 个小时来完成这个工作,并使用这个网站的帮助让代码可以工作,但我认为我这样做的方式是一种作弊方式,我使用了 scanf 命令。无论如何,我想以正确的方式解决这个问题。多谢你们!哦,代
我需要一些帮助来解决问题。 我有这个文本文件: 我将文本内容输入到字符串二维数组中,并将其转换为整数二维数组。当我转换为 int 数组时,nan 被替换为零。现在,我继续查找二维数组中每行的最大值和最
假设我有一个只能移动的类型。我们停止现有的默认提供的构造函数,但 Rvalue 引用引入了一种新的“ flavor ”,我们可以将其用于签名的移动版本: class CantCopyMe { priv
假设我有两个简单的对象,我想创建第三个对象来连接它们的属性。这非常有效: (()=>{ const a1 = {a: 2, b: 3} const b1 = {a: 100, c: 5}
我想知道我是否可以稍后在这样的代码中为 VAR 赋值 var myView: UIView func createView() { myView = UIView() { let _view =
我遇到了一些 Javascript/HTML/CSS 代码的问题。我对创建网站还很陌生,所以请多多包涵。 我最终想做的是从 javascript 中提取一个动态值并使用它对一些 div(在容器中)进行
#include class Box{ public: int x; Box(){ x=0; std::cout No move construction thanks to RV
我发现在javascript中&=运算符是按位赋值: var test=true; test&=true; //here test is an int variable javascript中是否存在
请帮助完成赋值重载函数的执行。 这是指令: 赋值运算符 (=),它将源字符串复制到目标字符串中。请注意,目标的大小需要调整为与源相同。 加法 (+) 和赋值 (=) 运算符都需要能够进行级联运算。这意
我有一个名为 SortedArrayList 的自定义结构它根据比较器对其元素进行排序,我想防止使用 operator[] 进行分配. 示例: 数组列表.h template class Array
我是 python 的新手,我看到了这种为列表赋值的形式 color= ['red' if v == 0 else 'green' for v in y] 但是如果我尝试用 3 个数字来做,例如 co
我是一名优秀的程序员,十分优秀!