- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
想象一个 TypeScript (4.2.2) 函数接收 string
或 Promise<string>
并使用它最初收到的相同类型返回一个答案,例如:
function trim(textOrPromise) {
if (textOrPromise.then) {
return textOrPromise.then(value => result.trim());
}
return textOrPromise.trim();
}
我想使用泛型定义此类函数的签名,以指示传入
Promise
总是导致
Promise
,并传入
string
结果是
string
为此,我定义了以下重载:
function trim(text: Promise<string>): Promise<string>
function trim(text: string): string
function trim(text: any): any {
if (text.then) {
return text.then(result => result.trim()); // returns Promise<string>
}
return text.trim(); // returns string
}
只要参数明确定义为
string
就可以正确转换或
Promise<string>
:
let value: string
trim(value) // fine
let value: Promise<string>
trim(value) // fine
但是,当我使用
union type 定义参数时(
Promise<string> | string
):
let value: Promise<string> | string
trim(value) // error: TS2769
我收到以下转译错误:
TS2769: No overload matches this call.
Overload 1 of 2, '(text: Promise<string>): Promise<string>', gave the following error.
Argument of type 'string | Promise<string>' is not assignable to parameter of type 'Promise<string>'.
Type 'string' is not assignable to type 'Promise<string>'.
Overload 2 of 2, '(text: string): string', gave the following error.
Argument of type 'string | Promise<string>' is not assignable to parameter of type 'string'.
Type 'Promise<string>' is not assignable to type 'string'.
有趣的是,当我将联合类型添加到函数签名时,该示例会正确地进行转换和行为
function trim(text: Promise<string>): Promise<string>
function trim(text: string): string
function trim(text: Promise<string> | string): Promise<string> | string
function trim(text: any): any {
if (text.then) {
return text.then(result => result.trim());
}
return text.trim();
}
let value: Promise<string> | string
trim(value) // fine
使用最后一个实现,TypeScript 知道当函数收到
Promise
时它将返回
Promise
,当它收到
string
它返回一个
string
.与
Promise<string> | string
的联合相反正如第三次重载所暗示的那样。
最佳答案
将联合传递给过载
在根据您的重载签名检查联合之前,Typescript 无法“拆分”联合。它正在检查您的 Promise<string> | string
类型的变量单独对抗每个过载。联合不能分配给它的任何一个成员,所以没有接受 Promise<string> | string
的重载签名。 .
这是一个已知的行为,并且可以追溯到几年前有关此的一些 GitHub 问题。
function trim(text: Promise<string>): Promise<string>;
function trim(text: string): string;
function trim(text: Promise<string> | string): Promise<string> | string;
function trim(text: Promise<string> | string): Promise<string> | string {
if (typeof text === "string") {
return text.trim();
} else {
return text.then(result => result.trim());
}
}
泛型
extends
包括工会。
T extends A | B
意味着
T
可以
A
,
B
,或工会
A | B
(或这些的任何更精致的版本,我稍后会介绍)。
function trim<T extends Promise<string> | string>(text: T): T {
然而,泛型在函数的实现方面会提出自己的问题,因为改进了变量
text
的类型。不细化泛型
T
的类型.考虑到类型
T
,这种行为通常是有意义的。可以是工会。
text
是
string
那么我们知道
T
必须包括
string
所以返回
string
应该没问题, 对?不。
string
、
Promise<string>
、
Promise<string> | string
)。但是
extends
为
T
打开无限可能的类型那是那些的改进。如
T
是文字串
" A "
然后是
string
"A"
我们从
text.trim()
返回真的不能分配给
T
.所以我们解决了一个问题,但又创造了另一个问题。我们必须做
as
断言以避免错误,例如:
Type 'string' is not assignable to type 'T'. 'string' is assignable to the constraint of type 'T', but 'T' could be instantiated with a different subtype of constraint 'string | Promise'.(2322)
function trim<T extends Promise<string> | string>(text: T): T {
if (text instanceof Promise) {
return text.then((result) => result.trim()) as T;
}
return (text as string).trim() as T;
}
const a = trim(' ' as Promise<string> | string); // type: string | Promise<string>
const b = trim(' '); // type: ' ' -- actually wrong!
const c = trim(' ' as string); // type: string
const d = trim(new Promise<string>(() => ' ')); // type: Promise<string>
const e = trim(new Promise<' '>(() => ' ')); // type: Promise<' '> -- wrong again!
Typescript Playground Link
关于typescript - 重载签名、联合类型和 "No overload matches this call"错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66507585/
假设我有一个类,我在其中重载了运算符 == : Class A { ... public: bool operator== (const A &rhs) const; ... };
我知道你不应该使用 std::find(some_map.begin(), some_map.end()) 或 std::lower_bound,因为它会采用线性时间而不是 some_map.lowe
我正在尝试在 Haskell 中定义 Vector3 数据类型,并允许在其上使用 (+) 运算符。我尝试了以下方法: data Vector3 = Vector3 Double Double Doub
我已经为我的类图将运算符重载为“-”。它的用途并不完全直观(糟糕的编码 - 我知道)但是如果我做 graph3 = graph2-graph1 那么图 3 是应该只接收图 2 和图 1 中的那些顶点。
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Operator overloading 我想重载 以按字母顺序排列字符串,但我不确定该怎么做。 如何再次
下面的代码给我一个编译错误。谁能告诉我为什么? class mytype { public: int value; mytype(int a) { value = a;
这有什么问题吗? class Vec2 attr_accessor :x, :y # ... def += (v) @x += v.x @y += v.y retu
是否可以重载 [] 运算符两次?允许这样的事情:function[3][3](就像在二维数组中一样)。 如果可能的话,我想看看一些示例代码。 最佳答案 您可以重载 operator[] 以返回一个对象
我的团队目前正在与 Lua 合作,创建一个 android 游戏。我们遇到的一件事是表面上无法创建重载构造函数。 我习惯于使用默认值设置一个对象,然后在需要时使其过载。 前任: apples() {
我有一个网页,在某个时候显示一个导航栏,它只不过是一个 a 元素的列表 (ul)。所述 a 元素的大多数样式规则都是通用的。唯一应该改变的部分是要显示的图像,可以从列表中每个 li 元素的 id 标签
我对使用/重载“范围步长”运算符(.. ..)很感兴趣,但我终其一生都无法了解如何使用它。 在文档中它说 // Usage: start .. step .. finish 但是在 F# shell
Java 11(可能无关紧要): public static String toString(Object obj) { return ReflectionToStringBuilder.to
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
我无法理解以下代码(针对行号进行注释) class Base { void m1(Object o) { } void m2(String o) { } } publi
我有以下代码片段: #include using namespace std; struct Integer{ int x; Integer(const int val) : x(v
class myclass{ //definitions here }; myclass e; int myarray[10]; /* Do something... */ e = myarray;
为什么不能将下标运算符(operator [])作为 friend 函数重载? 最佳答案 正如Bjarne Stroustrup在D&E book中所说: However, even in the o
我有以下代码片段: #include using namespace std; struct Integer{ int x; Integer(const int val) : x(v
因此,我有一个问题是我最近尝试重载 namespace Eng { /** * A structure to represent pixels */ typedef
如何重载onResume()以正确的方式工作?我想从 activity 返回到 MainActivity ,我希望在其中具有与应用程序启动后相同的状态。我想使用 recreate() 但它循环了或者类
我是一名优秀的程序员,十分优秀!