- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
谁能帮我理解这里发生了什么:TS playground
基本上我有一个 store
有一个 exec
方法,我想为子进程缩小 exec
参数的类型但是好像有错误,store类型是泛型
type Param<Options> = {
[K in keyof Options]: Readonly<{
id: K,
options: Options[K],
}>
}[keyof Options];
interface Store<Options> {
exec: (nextState: Param<Options>) => void
}
type ParentOptions = {
'a': { a: string },
} & SubOptions
type SubOptions = {
'b': { b: number },
}
function test(
parentFlowExec: (nextState: Param<ParentOptions>) => void,
subFlowExec: (nextState: Param<SubOptions>) => void,
parentNonGeneric: { exec: (nextState: Param<ParentOptions>) => void },
subNonGeneric: { exec: (nextState: Param<SubOptions>) => void },
parentFlow: Store<ParentOptions>,
subFlow: Store<SubOptions>,
) {
parentFlowExec = subFlowExec; // error: ok
subFlowExec = parentFlowExec; // passed
parentNonGeneric = subNonGeneric; // error: ok
subNonGeneric = parentNonGeneric; // passed
parentFlow = subFlow; // error: ok
subFlow = parentFlow; // error ??
// I plan to use it like this
subProcess(parentFlow);
}
function subProcess(flowStore: Store<SubOptions>) {
flowStore.exec({ id: 'a', options: { a: 'a' } }); // can't call with 'a'
flowStore.exec({ id: 'b', options: { b: 3 } }); // ok
}
更新:我将参数移出并得到 it working但仍然不明白为什么嵌套它们不起作用
interface Store<Options> {
exec: (nextState: Options) => void
}
// parent2: Store2<Param<ParentOptions>>,
// sub2: Store2<Param<SubOptions>>,
最佳答案
为了回答您的问题,首先,让我们快速回顾一下不同“方差”的含义。在下表中,我使用了 Microsoft 的 .NET documentation 中的定义。 (文档中没有的双方差除外)因为我发现它们最容易掌握:
让我们检查一下您的类型中哪些是父类(super class)型,哪些是子类型:
type T1 = SubOptions extends ParentOptions ? true : false; // false
type T2 = ParentOptions extends SubOptions ? true : false; // true
接下来是 PartentOptions
是 SubOptions
的子类型,而后者是它的父类(super class)型。它告诉我们什么?它告诉我们,当你注释 subFlow
作为Store<SubOptions>
然后尝试分配 parentFlow
对于它(注释为 Store<ParentOptions>
),您正在尝试分配一个 子类型,其中需要一个父类(super class)型。
如果我们引用方差表,我们会看到这需要协方差,但是当您收到错误时,这意味着我们正在处理逆变 或 < em>不变性。现在,当您分配 subFlow
至 parentFlow
,您正在分配一个需要子类型的父类(super class)型。
上面也会导致错误,意味着这里的赋值实际上是不变,并且@captain-yossarian的 comment是正确的:
I believe that it is because subFlow and parentFlow are invariant to each other.
然而,这种行为是 TypeScript 的设计限制(参见 Anders Hejlsberg 的 comment 相关问题)牺牲了一些灵 active 以保证稳健性(删除 [keyof Options]
索引,您将看到逆变赋值成为可能) .
至于您的解决方案,由于方差分析的工作原理,当您移动 Params
时向外,参数类型变为协变(因为 T[keyof T]
在这里没有别名。请注意,当简化为裸结构时, Param
类型正是: type Param<Options> = Options[keyof Options]
,仅映射1).
看一下您的解决方案的简化示例0:
type Param<Options> = {
[K in keyof Options]: Readonly<{
id: K,
options: Options[K],
}>
}[keyof Options];
interface Store<Options> {
exec: (nextState: Options) => void
}
type SuperOptions = { 'b': { b: number } }
type SubOptions = { 'a': { a: string } } & SuperOptions
const test1 = (subtype: Store<Param<SubOptions>>) => subProcess1(subtype); // OK, Subtype -> Supertype, covariance
const test2 = (supertype: Store<Param<SuperOptions>>) => subProcess2(supertype); // error, Supertype -> Subtype, contravariance
const subProcess1 = (supertype: Store<Param<SuperOptions>>) => supertype.exec({ id: 'b', options: { b: 3 } }); // ok
const subProcess2 = (subtype: Store<Param<SubOptions>>) => subtype.exec({ id: 'b', options: { b: 3 } }); // ok
0 你的命名选择稍微增加了一个已经很棘手的问题的困惑:一个子类型被称为 ParentOptions
和父类(super class)型 SubOptions
,而它们之间的关系是相反的,所以我将它们命名为SubOptions
。和 SuperOptions
相应地使事情更清楚。
1 从评论中的讨论中,必须注意的是 Store<Param<SubOptions>>
之间的关系和 Store<Param<SuperOptions>>
在解决方案中是协变,T[keyof T]
这是逆变(参见 Anders 的 comment - SuperOptions
父类(super class)型的属性少于 SubOptions
子类型,并且没有判别式)。
关于带有一般错误的 typescript 缩小类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67482793/
我想使用 li 和 ul 制作一个多级下拉列表,以便显示我博客中按年和月排序的所有文章。我希望我的下拉菜单看起来像 Google Blogspot 下拉菜单: 这是我的 CSS 和 HTML 代码 u
我在 Win 7 64 机器上将 CodeBlocks 与 gcc 4.7.2 和 gmp 5.0.5 结合使用。开始使用 gmpxx 后,我看到一个奇怪的段错误,它不会出现在 +、- 等运算符中,但
我正在使用 tern 为使用 CodeMirror 运行的窗口提供一些增强的智能感知,它工作正常,但我遇到了一个问题,我想添加一些自定义“types”,可以这么说,这样下拉列表中它们旁边就有图标了。我
我正在尝试让我的 PC 成为 Android 2.3.4 设备的 USB 主机,以便能够在不需要实际“附件”的情况下开发 API。为此,我需要将 PC 设置为 USB 主机和“设备”(在我的例子中是运
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
我在设置服务器方面几乎是个新手,但遇到了一个问题。我有一个 Ubuntu 16.04 VPS 并安装了 Apache2 和 Tomcat7。我正在为 SSL 使用 LetsEncrypt 和 Cert
我在一个基于谷歌地图的项目上工作了超过 6 个月。我使用的是 Google Maps API V1 及其开发人员 API key 。当我尝试发布应用程序时,我了解到 Google API V1 已被弃
我是 Python 的新手,所以如果我对一些简单的事情感到困惑,请原谅。 我有一个这样的对象: class myObject(object): def __init__(self):
这个问题已经有答案了: How can I access object properties containing special characters? (2 个回答) 已关闭 9 年前。 我正在尝
我有下面的 CSS。我想要的是一种流体/液体(因为缺乏正确的术语)css。我正在为移动设备开发,当我改变模式时 从纵向 View 到陆地 View ,我希望它流畅。现在的图像 在陆地 View 中效
我正在尝试使用可以接受参数的缓存属性装饰器。 我查看了这个实现:http://www.daniweb.com/software-development/python/code/217241/a-cac
这个问题在这里已经有了答案: Understanding slicing (36 个答案) 关闭 6 年前。 以a = [1,2,3,4,5]为例。根据我的直觉,我认为 a[::-1] 与 a[0:
mysqldump -t -u root -p mytestdb mytable --where=datetime LIKE '2014-09%' 这就是我正在做的事情,它会返回: mysqldum
我正在制作销售税计算器,除了总支付金额部分外,其他一切都正常。在我的程序中,我希望能够输入一个数字并获得该项目的税额我还希望能够获得支付的总金额,包括交易中的税金。到目前为止,我编写的代码完成了所有这
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许在 Stack Overflow 上提出有关通用计算硬件和软件的问题。您可以编辑问题,使其成为
我是否必须进行任何额外的设置才能让 apache-airflow 在任务失败时向我发送电子邮件。我的配置文件中有以下内容(与默认值保持不变): [email] email_backend = airf
这个问题在这里已经有了答案: What does the $ symbol do in VBA? (5 个回答) 3年前关闭。 使用返回字符串(如 Left)的内置函数有什么区别吗?或使用与 $ 相同
我有一个用VB6编写的应用程序,我需要使用一个用.NET编写的库。有什么方法可以在我的应用程序上使用该库吗? 谢谢 最佳答案 这取决于。您可以控制.NET库吗? 如果是这样,则可以修改您的库,以便可以
当我创建一个以 ^ 开头的类方法时,我尝试调用它,它给了我一个错误。 class C { method ^test () { "Hi" } } dd C.new.test; Too m
我已经使用 bower 安装了 angularjs 和 materialjs。 凉亭安装 Angular Material 并将“ngMaterial”注入(inject)我的应用程序,但出现此错误。
我是一名优秀的程序员,十分优秀!