|
|
|
|
|
by jakear
1204 days ago
|
|
Not quite, TypeScript provides a options beyond what the author of this article details that IMO are superior, at least in some cases. Instead of just "throw an error or return ()" or "throw an error or return NonEmpty<T>", you can declare a function's return type as "throws iff the argument isn't NonEmpty" or "true iff the argument is NonEmpty". Compare: function validateNonEmpty<T>(list: T[]): void {
if (list[0] === undefined)
throw Error("list cannot be empty")
}
function parseNonEmpty<T>(list: T[]): [T, ...T[]] {
if (list[0] !== undefined) {
return list as [T, ...T[]]
} else {
throw Error("list cannot be empty")
}
}
function assertNonEmpty<T>(list: T[]): asserts list is [T, ...T[]] {
if (list[0] === undefined) throw Error("list cannot be empty")
}
function checkEmptiness<T>(list: T[]): list is [T, ...T[]] {
return list[0] !== undefined
}
declare const arr: number[]
// Error: Object is possibly undefined
console.log(arr[0].toLocaleString())
const parsed = parseNonEmpty(arr)
// No error
console.log(parsed[0].toLocaleString())
if (checkEmptiness(arr)) {
// No error
console.log(arr[0].toLocaleString())
}
assertNonEmpty(arr)
// No error
console.log(arr[0].toLocaleString())
For me the `${arg} is ${type}` approach is superior as you are writing the validation once and can pass the precise mechanism for handling of the error to the caller, who tends to have a better idea of what to do in degenerate cases (sometimes throwing a full on Exception is appropriate, but sometimes a different form of recovery is better). |
|
This is quite nice in situations where the type system already supports the refinement in question (which is true for this NonEmpty example), but it stops working as soon as you need to do something more complicated. I think sometimes programmers using languages where the TS-style approach is idiomatic can get a little hung up on that, since in those cases, they are more likely to blame the type system for being “insufficiently powerful” when in fact it’s just that the convenience feature isn’t sufficient in that particular case. I presented an example of one such situation in this followup blog post: https://lexi-lambda.github.io/blog/2020/08/13/types-as-axiom...