4

Typescript 是否支持直接测试结构类型兼容性?

C# 支持is运算符和类型表面IsAssignableFrom(object instance)

if (foo is SomeType) ...
if (SomeType.IsAssignableFrom(foo)) ...

是否有一些直接的方法可以在 Typescript 中执行这种检查,或者我是否必须探测每个必需的成员?

instanceof可能会为手头的情况做,但不尊重结构兼容性,而是检查原型链。

是的,我知道,Typescriptinstanceof直接相当于 C# 的is运算符。但我确实从指定结构类型开始。

也许我应该把一个庸医的方法放在Object

4

1 回答 1

2

我相信这里的适当答案是

不是,但!

所以只是为了解决这个问题,Typescript 的类型信息只存在于编译时。无法在运行时检查结构相等性。您当然可以在编译时进行此检查,但这只是称为分配,可能不是您要问的。

长话短说,您必须(通常)探查所需的成员,但有几种方法可以巧妙地解决这个问题。


从 Typescript 1.6 开始,该语言支持用户定义的类型保护

这些作为便利功能非常棒,并且是一种安全地向下转换/指定/优化类型的方法(取决于您喜欢的术语)。如果您希望以典型的方式(.hasOwnProperty等等)手动检查某些类型的存在,根据我的经验,它们作为发送消息类型的方式更有效。在大多数消息传递系统中,您可能有一个类型属性。你可以定义一个函数

function isConfigGetMsg(msg: BaseMessage): msg is ConfigGetMsg {
  return msg.name === MsgTypes.CONFIG_GET_MSG;
}

//and use it as
if (isConfigGetMsg(msg)){
    console.log(msg.key); //key being a ConfigGetMsg only value
}

只要您的消息格式正确,这就可以了。尽管如您所见,您需要测试特定信息。如果您想要更通用的方法,这不是它。


正如您所提到的,您可以探测每个必需的成员。你可以通过使用前面的技术来创建这样的东西来让自己更容易一些:

interface One {
    a: number
}
interface Two extends One {
    b: string
}
function isAssignableTo<T>(potentialObj:Object, target:T) : potentialObj is T {
    return Object.keys(target).every(key => potentialObj.hasOwnProperty(key))
}

let a: Object = {
    a: 4,
    b: 'hello'
};

let b: Two = undefined;

if(isAssignableTo(a, b)) {
    b = a;
}

功能是这里的isAssignableTo重点。参数的方向是有争议的,但也不相关,所以你可以随意放置它们。这不是一个理想的解决方案。您可能希望仅基于非功能属性来分配事物,例如,暗示状态而不是通用功能。您可以尝试扩展检查以解决此问题,但无论如何都存在陷阱。


我很抱歉地说我相信唯一的“真实”答案是一个还不可行的答案。如果您正在使用类,您可以使用装饰器来捕获有关您的应用程序的元数据,包括以后可以用于反射/检查的类型信息。您可以创建自己的装饰器来执行此操作,但您也可以利用typescript 自己的元数据生成来生成类属性。您可以使用 tsconfig 标志打开。

这假设存在元数据反射 api。您需要使用它来提取该信息。然后,您可以重写之前的 isAssignableTo 函数以处理元数据。如果我没记错的话,这仍然会遇到类型存储为字符串的问题,如果你有子类型,它们将不匹配。


无论如何,我希望这会有所帮助。对不起,我不能给你一个简单的“是”。知道我的运气你实际上只是在寻找任务;)

于 2016-05-17T06:45:31.917 回答