- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我有一个系统用于通过套接字连接来回传递 JSON 消息。它使用标记联合作为消息类型:
export type ErrorMessage = { kind: 'error', errorMessage: ErrorData };
export type UserJoined = { kind: 'user-joined', user: UserData };
// etc
export type Message = ErrorMessage | UserJoined | /*etc*/;
它在基本代码中运行得相当好,但我有一个构建在它之上的模块,我想扩展代码。我有一个新的消息类型要添加:
export type UserAction = { kind: 'user-action', action: Action }
这里的问题是我无法扩展“消息”以将我的新 UserAction 包含到联合中。我想我可以制作自己的扩展消息:
export type ExtendedMessage = Message | UserAction;
但这里的问题是这看起来很笨重,第一。我无法将我的新 UserAction 传递到任何需要消息的方法中,即使代码实际上应该可以正常工作。以后想要同时扩展我的模块和基本模块的任何其他人都需要创建第三种类型:export type ExtendedMessageAgain = ExtendedMessage | MyNewMessage
.
所以。我已经看到通过添加新的 .d.ts 文件扩展了其他属性的接口(interface)(比如 passport 如何扩展 Express JS 的 Request 对象以添加身份验证属性),我认为标记联合也必须存在类似的东西,对吧?
但似乎并非如此。我到处搜索,没有看到任何地方使用这种模式。这让我相信,也许我的设计在某种程度上是错误的。但我没有办法解决它。
我不想使用类,因为类型信息通过网络被删除; kind
属性必须存在。我喜欢这种范式:
declare var sendMessage = (message: Message) => void;
sendMessage( { kind: 'error', errorMessage: { /* */ } }); // ok
sendMessage( { kind: 'random', parameter: { /* */ } }); // error, no kind 'random'
sendMessage( { kind: 'error', message: { /* */ } }); // error, no property 'message' on 'error'
但我看到的唯一解决方案是使 Message
成为接口(interface)基础,如下所示:
export interface Message { kind: string }
export interface ErrorMessage extends Message { errorMessage: ErrorData }
declare var sendMessage = (message: Message) => void;
sendMessage( { kind: 'error', errorMessage: { /* */ } }); // ok
sendMessage( { kind: 'random', parameter: { /* */ } }); // ok
sendMessage( { kind: 'error', message: { /* */ } }); // ok
而且这个方法失去了上面所有的类型保护。
那么...有没有办法在多个模块中扩展 Tagged Union,影响类型的原始名称,而不定义新类型?还是这里有我只是没有看到的更好的设计?
这是启发这篇文章的代码:https://github.com/RonPenton/NotaMUD/blob/master/src/server/messages/index.ts
我正在寻求大规模重构,以便我可以将所有消息移出到单独的模块中,而不是随着时间的推移这个文件变得一团糟。
最佳答案
您可以这样做来定义 Message
的联合类型:
export interface MessageTypes {}
export type Message = MessageTypes[keyof MessageTypes]
然后无论您在哪里定义新的消息类型,都执行以下操作:
export type UserAction = { kind: 'user-action', action: Action }
declare module '../message' { // Where you define MessageTypes
interface MessageTypes {
UserAction: UserAction
}
}
因此 MessageTypes
接口(interface)的值成为联合类型,您可以使用声明合并向接口(interface)添加更多值,这将自动更新联合类型。
您可以查看 TS 文档以获取有关声明合并的更多信息:https://www.typescriptlang.org/docs/handbook/declaration-merging.html
关于typescript - 在不同的模块中扩大 TypeScript 中的标记/区分联合?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46392758/
我正在通过修改我为处理 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、价格
我是一名优秀的程序员,十分优秀!