- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在 TypeScript 中,我想创建一个联合类型来表示属于一个或多个不同类型的值,类似于 oneOf
在 OpenAPI或 JSON Schema .根据a previous answer on a similar question ,TypeScript 中的联合运算符应该执行此操作,因为它表示集合联合(包含或)。但是,当 using the in operator as a type guard 时,这与 TypeScript 中的类型推断行为不匹配。 ,它似乎假设值完全属于联合类型之一(析取联合或异或)。作为一个简单的例子,这个 leads to a type error :
interface SoftwareDeveloper { code(): void; }
interface Chef { cook(): void; }
type SoftwareDeveloperOrChef = SoftwareDeveloper | Chef;
// Determined at runtime. May have the code method, the cook method, or both.
const person = { code: () => { }, cook: () => { } };
const softwareDeveloperOrChef: SoftwareDeveloperOrChef = person;
if ("code" in softwareDeveloperOrChef) {
// softwareDeveloperOrChef has inferred type SoftwareDeveloper
softwareDeveloperOrChef.code();
// Coding is hungry work
if ("cook" in softwareDeveloperOrChef) {
// Not allowed, since softwareDeveloperOrChef has inferred type never
softwareDeveloperOrChef.cook();
}
}
为了获得我想要的行为和avoid type errors ,我必须手动将交集类型的所有组合添加到联合中:
type SoftwareDeveloperOrChefOrBoth = SoftwareDeveloper | Chef | (SoftwareDeveloper & Chef);
const softwareDeveloperOrChefOrBoth: SoftwareDeveloperOrChefOrBoth = person;
if ("code" in softwareDeveloperOrChefOrBoth) {
// softwareDeveloperOrChef has inferred type SoftwareDeveloper | (SoftwareDeveloper & Chef)
softwareDeveloperOrChefOrBoth.code();
if ("cook" in softwareDeveloperOrChefOrBoth) {
// Allowed, since softwareDeveloperOrChefOrBoth has inferred type SoftwareDeveloper & Chef
softwareDeveloperOrChefOrBoth.cook();
}
}
一个中间问题可能是这是否是我应该期望的行为?但是,考虑到它是已实现的行为,我实际上对为任意数量的类型构造非析取联合类型的方法更感兴趣。随着类型数量的增加,手动执行此操作会导致联合类型定义的大小呈指数增长:
type AorB = A | B | A & B;
type AorBorC = A | B | C | A & B | A & C | B & C | A & B & C;
type AOrBOrCorD = A | B | C | D | ... | B & C & D | A & B & C & D;
我可以为特定数量的参数编写通用类型:
type AnyOf2<A, B> = A | B | A & B;
type AnyOf3<A, B, C> = AnyOf2<A, AnyOf2<B, C>>;
type AnyOf4<A, B, C, D> = ...;
但是我可以使用类似于 AnyOf<A | B>
中使用的技术来制作一个采用任意数量类型的类似泛型类型,例如联合 ( UnionToIntersection
)实用类型 implemented here ?
最佳答案
出现错误的原因是使用联合类型不能正确地允许 SoftwareDeveloperOrChef
两者都是 SoftwareDeveloper
和 Chef
.如果我们知道对象是 SoftwareDeveloper
那么我们根本没有关于 cook
的任何信息属性(property)——可以是任何东西。它不必是预期类型的函数。
您的 type AnyOf2<A, B> = A | B | A & B;
就快完成了,但是当我们处理对象的属性时,我们可以使用 Partial
和 Required
以获得更好的类型。
我们希望为 SoftwareDeveloperOrChefOrBoth
实现的最终类型是cook
和 code
都是可选函数,但至少有一个是必需的。
试试这个:
type EitherOrBoth<A, B> = Partial<A & B> & (A | B);
type SoftwareDeveloperOrChef = EitherOrBoth<SoftwareDeveloper, Chef>;
一个变化是使用Partial
你不能再使用 "code" in softwareDeveloperOrChef
作为类型保护,因为对象可以将属性键显式设置为 undefined
.例如{code: undefined; cook: () => {};}
是我们类型的有效成员。
但是这段代码 check out
const softwareDeveloperOrChef: SoftwareDeveloperOrChef = person;
if (softwareDeveloperOrChef.code) {
softwareDeveloperOrChef.code();
if (softwareDeveloperOrChef.cook) {
// this is allowed now
softwareDeveloperOrChef.cook();
}
}
编辑
根据我们在评论中的讨论,您需要单独的接口(interface) SoftwareDeveloper
和 Chef
有多种方法,您希望能够确定 Chef
中的一种方法是否有效存在,Chef
的所有其他方法也将出席。
为了最大限度的类型安全,组合 SoftwareDeveloperOrChefOrBoth
type 应该明确声明任何成员必须拥有 Chef
的所有方法或者没有。
此实用程序类型 Not<T>
通过说 T 的任何键不能设置为 undefined
以外的任何值来处理此问题:
export type Not<T> = {
[P in keyof T]?: never;
};
请注意,仅不包括该属性并不表示无法设置它。如 linked thread 中所述, 和你的 type AorB
当创建一个新的无效对象(一个具有 SoftwareDeveloper
的所有方法和一些但不是所有 Chef
的方法)时,您会得到一个错误,但在将类型分配给现有对象时不会,因为这个无效对象可分配给A
因此也可以分配给 AorB
.
而不是宽松的 EitherOrBoth<A,B>
之前建议的类型,这里是一个更严格的强制排他性的类型:
type ExclusiveAndOr<A, B> = A & Not<B> | B & Not<A> | A & B;
type SoftwareDeveloperOrChef = ExclusiveAndOr<SoftwareDeveloper, Chef>;
有了这个类型,就不再只是假设一个人可以cook
他们也可以chop
.
function checkPerson( softwareDeveloperOrChef: SoftwareDeveloperOrChef ) {
if (softwareDeveloperOrChef.code) {
softwareDeveloperOrChef.code();
if (softwareDeveloperOrChef.cook) {
// should be ok
softwareDeveloperOrChef.cook();
// ok to assume all methods exist
softwareDeveloperOrChef.chop();
} else {
// should be error
softwareDeveloperOrChef.chop();
}
}
}
关于typescript - TypeScript 中任意数量类型的非析取联合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62698675/
我正在通过修改我为处理 slice 而创建的库来玩转泛型。我有一个 Difference接受 slice 并返回仅在其中一个 slice 中找到的唯一元素列表的函数。 我修改了函数以使用泛型,并且我正
Typescript 编译器 在我尝试使用联合或多个类型/接口(interface)时不断抛出错误。 My requirement 我从服务器收到一个对象作为响应,其中一个键 ('errorMessa
我需要在 SQLAlchemy 中执行 2 选择。例如: select1 = Session.query(col1, col2, col3, col4).filter(...) select2 = S
我建立了一个数据库来输入我所有的头痛和偏头痛跟踪数据。我正在提取一些查询,这些查询显示某一年中按月计算的不同头痛严重程度的计数。我有一个查询按月得到所有头痛,另一个在一定严重程度下得到头痛,最后一个在
我有三个表,一个是默认值表。 我需要做的是选择 TableA 和 TableB 的值,并从默认值的选择中回填任何缺失的值。 每个表都有一个键和值列。 数据的一个例子可能是这样的: DefaultTab
我正在尝试构建一个 单个 JSONPath 查询 ,它将测试 是否存在两个或多个路径 。 让我们考虑以下示例文档: { "firstName": "John",
我正在尝试基于对象中的嵌套属性创建联合类型。请参见下面的示例: type Foo = { abilities: { canManage: boolean } } typ
我有以下查询: SELECT result.globalId AS id, result.date, p1.playerName AS player, p2.playerName AS targe
我有两张 table 。第一个每天刷新。(该表有超过 10 列,但其中 2 列是相关的)我想根据 vid (这是一个唯一的 id )和人口进行每日统计。新的视频 ID 每天都会出现和消失。例如: 第一
这个问题已经有答案了: How to know what table a result came from when using UNION in MySQL (1 个回答) 已关闭 6 年前。 让我
我有 2 个表,一个列出人员及其与其属性的关系,另一个表列出属性(名字、姓氏等)。 人员表中的每个人可能不具有属性表中列出的所有属性。我想要的是每个人都为每个属性返回一行,无论他们是否有链接。 举个例
假设我们有 MySQL 服务器 A,我们需要在其中创建位于服务器 B 上的表的“副本”。 我们没有启用联合。重置服务器 A 会造成很多麻烦,我相信,我们不能在不重置的情况下启用联合。我也认为在B服务器
我有一个 Java 类 A。A 的构造函数调用了几个方法 m1、m2。 class A{ public A(){ m1(); m2(); ......
我正在开发一种编程语言,我想为其提供一个Range 数据类型,目前它不是通常的int 对列表。值 (x,y)约束条件是 x < y .我说不像通常那样,因为通常一个范围只是一对,但在我的例子中,它超过
我正在寻找加速一段合并两个 SortedLists 的代码。 C# 4.0 通用 SortedList:http://msdn.microsoft.com/en-us/library/ms132319
如果我有以下包含函数及其参数的联合,我该如何调用它? type Wrapper = { fn: (a: string) => void arg: string } | { fn: (a:
我正在尝试移植一个内部有一个联合的 C 结构。 Winapi.Winsock2.pas 中的默认结构记录中缺少某些字段。 但这是正确的方法吗?谢谢。 typedef struct _WSACOMPLE
我希望通过“版本”编号的前 8 个字符的子字符串对以下查询的结果进行排序。我理解 SUBSTRING(),所以不要用这个来打扰我。我的问题是尝试实际放置关于 UNION 的 ORDER BY。 更新:
我需要创建一个带有联合的 QueryBuilder,这可能吗? $qb = $this->em->createQueryBuilder() ->select('table1.numObject
我正在为 Magic the Gathering Cards 创建库存系统,需要使用主要卡片信息更新价格。 我有两个表,卡片和价格 卡片有以下列:ID、姓名、Ed、价格 价格有以下列:姓名、Ed、价格
我是一名优秀的程序员,十分优秀!