gpt4 book ai didi

typescript - 安全地消除 typescript 中的无效性

转载 作者:行者123 更新时间:2023-12-05 09:32:13 27 4
gpt4 key购买 nike

我正在寻找有关处理空值的更好方法的建议。

我目前正在学习 typescript 。我经常遇到的问题之一是将可能未定义或 null 的值转换为既不是 undefined 也不是 null 的类型化值。

例如

    currentTarget?: HTMLEement;

...
let position = this.currentTarget.styles.position;

这当然会产生错误,因为 currentTarget 可能是 undefined。这很棒。我明白了。

通常,人们通过间接方式知道可能为 null 的表达式一定不能为 null。在这种情况下,如果表达式实际上为 null,我更倾向于抛出错误,而不是使用 null 合并运算符或保护 if 语句来使错误消失。例如,我可以编写 someObject?.someProperty?.someMethod() 来消除错误;但它忽略了这样一个事实,即如果这些空合并运算符中的任何一个有效,它几乎总是一个错误。最好(尽管不可能)编写 someObject.someProperty.someMethod() 并抛出异常。

我开始编写保护函数来消除空值。例如:

model?: Model;

getModel(): Model {
if (!this.model) throw new StateError("Model is null.");
return this.model;
}

并考虑过以下想法:

function nullCast<T>(value: T | null | undefined): T {
if (!value) throw new Error("Value is null");
return value;
}

[编辑:更正了 nullCast 的代码(效果非常好。]

没有什么可怕的

if (!this.model) throw new StateError("Model is null");
... carry on with non-nullish model...

而且我逐渐学会为变量提供默认值,即使它们是复杂类型,而不是将对象声明为可空值。

{
model: SomeModel.EmptyModel();
... instead of ...
model?: SomeModel;

}

但最后,似乎我的代码中不可忽略的部分正在处理可空变量,我知道我知道这些变量不是空的,因为我所在的函数不会执行如果程序处于这些变量可能具有空值或未定义值的状态。

目前手头的一个引人注目的例子:一个通过 websocket 从远程服务器获取初始状态的模型类。初始状态是异步填充的,在此期间显示“正在加载...”屏幕。这反过来会强制实现 UI 的代码处理 null 或未定义的模型属性,如果显示 UI,这些属性不可能为 null。

虽然我明白需要付出努力才能完全正确,但我不禁认为我缺少更好的方法来处理这个问题。对替代方案或最佳实践的建议表示赞赏。

最佳答案

"For example, I could write someObject?.someProperty?.someMethod() which makes the error go away; but it ignores the fact that it is almost always an error if any of those null coalescing operators have effect."

如果有一个空值存在异常情况,那么您的类型不应该允许该空值。听起来你的类型太宽容了。如果值为 null 时出错,那么按理说当值为 null 时应该是类型错误

处理空值是我们所有人都必须处理的事情,但它真的很难一概而论,因为在空值情况下要做什么在很大程度上取决于您的应用程序/代码的工作方式。


一般来说,当我拥有需要非空的属性时,我会这样声明它们。

interface MyType {
someObject: {
someProperty: {
someMethod(): void
}
}
}

然后仅在需要时才将字段设为可选:

function makeMyType(values: Partial<MyType>): MyType {
return { ...defaultMyType, ...values }
}

有时它真的无法轻易避免,在这种情况下我确实做了很多这样的事:

if (!foo) throw new Error('really expected foo to be here!')
console.log(foo.bar)

或者在某些情况下只是:

if (!foo) return
console.log(foo.bar)

这很好。如果它真的是无效的,那么您会收到一条很好的错误消息,说明出了什么问题。或者,在第二种情况下,您只需退出在 null 情况下无效的函数。


如果没有您的类型的详细信息,很难给出更具体的建议。但最后如果:

"it seems like a non-negligble portion of my code is dealing with nullish-able variables"

然后我会先查看您的类型,以确保您尽可能多地删除无效性。


另一种选择是使用 type predicate function验证您可以使用的整个对象,而无需在任何地方测试每个属性:

interface Foo {
foo: string
bar: string
baz: string
}

function validateFoo(maybeFoo: Partial<Foo>): maybeFoo is Foo {
return !!maybeFoo.foo && !!maybeFoo.bar && !!maybeFoo.baz
}

const partialFoo: Partial<Foo> = { bar: 'a,b,c' }
if (validateFoo(partialFoo)) {
partialFoo.bar.split(',') // works
}

Playground

关于typescript - 安全地消除 typescript 中的无效性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68293907/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com