AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 76923047
Accepted
land
land
Asked: 2023-08-17 23:37:46 +0800 CST2023-08-17 23:37:46 +0800 CST 2023-08-17 23:37:46 +0800 CST

为什么我们在条件语句中使用方括号“[]”?

  • 772

在这个例子中:

type Action<T extends string, P = never> = {
  type: T;
} & ([P] extends [never]
  ? {}
  : {
      payload: P;
    });

为什么我们需要在条件周围加上方括号[P] extends [never],如果我们不这样使用它有什么区别P extends never?

我明白这种类型在做什么,只是找不到解释为什么我们实际上需要方括号以及为什么没有方括号就不起作用。

编辑:让我尝试更简单的例子:


type GetTest<T = never> = [T] extends [never] ? string : number;

type Test = GetTest; // type Test = string; 

这里type Test = string正如预期的那样。但我没有得到的是,当我们没有方括号时,条件会发生什么,我们没有得到,string但我们也没有得到 a number,相反,我们得到了type Test = never

type GetTest<T = never> = T extends never ? string : number;

type Test = GetTest; // type Test = never

typescript
  • 1 1 个回答
  • 57 Views

1 个回答

  • Voted
  1. Best Answer
    jcalz
    2023-08-18T02:27:24+08:002023-08-18T02:27:24+08:00

    如果您有一个条件类型,其中正在检查的类型( 的左侧extends)是泛型类型参数,则它会成为分布式条件类型。所以在

    type GetTestDist<T = never> = T extends never ? string : number;
    type GetTestNonDist<T = never> = [T] extends [never] ? string : number;
    

    GetTestDist<T>是分布式条件类型,因为T是泛型类型参数,而GetTestNonDist<T>is 不是,因为[T]不是泛型类型参数。

    是的,[T] 涉及泛型类型参数T,但它本身不是类型参数。有时人们会说TinT extends never ? ⋯ : ⋯是裸类型参数或裸类型参数,而不是Tin [T] extends [never] ? ⋯ : ⋯,比如说“穿衣服的”。

    因此,所检查的类型是裸类型参数的条件类型是分布式条件类型。但是,这是什么意思?


    当类型函数F<T>是分配条件类型时,该操作将分布在 中的联合上T,这意味着它单独作用于每个联合成员并将结果连接回联合中。因此,例如,F<A | B | C>将评估为与 相同F<A> | F<B> | F<C>。

    此外,该never类型被认为是空联合,因此无论如何F<never>都将始终如此。never(有关权威来源,请参阅microsoft/TypeScript#23182 上的相关评论)。

    这种处理never可能会令人困惑,但它是一致的。该never类型被吸收到联合中:也就是说,X | never被简化为X无论X是什么。x(如果你有一个type 的值X | never,那么x要么是 type 的X值,要么是 type 的值,never但是没有type 的值,所以必须是 type 的。所以所有可分配给的值都必须可分配给,并且类型检查器通过吸收来观察这一点联合。有关更多信息,请参阅为什么永远不能分配给每种类型?)您可以将其视为联合操作的单位元素。neverx XX | neverXnevernever

    因此,如果X | never等于never,则F<X>和F<X | never>相同。如果F<T>是可分配的,则F<X | never>等价于F<X> | F<never>。这意味着对于所有人来说F<X>都是一样的。要实现这一点,最简单的方法是 if is just 。F<X> | F<never>XF<never>never

    (以此类推,假设您有一个数学函数 ?(?),它对加法具有分配性,因此 ?(?) + ?(?) = ?(?+?)。加法有一个单位元素:0,因为 ?+ 0 = ?。这意味着 ?(0) 必须为 0,因为 ?(?) + ?(0) = ?(?+0) = ?(?)。0 是“空总和”,并且是“空的和never”联盟”。)


    分配行为通常是人们所希望的,但有时却并非如此。因此,关闭这种行为的方法是用一些协变类型函数“覆盖”裸类型参数和您正在检查的类型C<T>。(有关方差的更多信息,请参阅TypeScript 中方差、协方差、逆变和双方差之间的差异)。

    type Distrib<T> = T extends U ? X : Y
    type NonDistrib<T> = C<T> extends C<U> ? X : Y
    

    您需要在 的两侧都执行此操作extends,以便进行相同的检查( ifC<T>是协变的TthenC<A> extends C<B>意味着A extends B)。任何协变“服装”都足够了。

    但是,如果您正在寻找,呃,“最简洁”的服装,以便可以节省尽可能多的击键次数,那么您应该使用[T]. 无论好坏,TypeScript 将数组类型视为协变(请参阅为什么 TypeScript 数组是协变的?),并且单元素元组是一种数组类型。与任何其他协变类型函数相比,没有什么比使用更“正确”的了[T],它只是最方便的。

    因此,如果您想关闭条件类型中联合的分配,通常建议您将两边都用⋯包裹extends起来:[]

    type Distrib<T> = T extends U ? X : Y
    type NonDistrib<T> = [T] extends [U] ? X : Y
    
    • 2

相关问题

  • Nestjs有关模块的问题

  • 如何获取文件的打字稿元数据?

  • 如何将数组转换为像这样的对象返回 const 类型?

  • “没有重载匹配”我的 Object.assign() 调用;我该如何修复它?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    使用 <font color="#xxx"> 突出显示 html 中的代码

    • 2 个回答
  • Marko Smith

    为什么在传递 {} 时重载解析更喜欢 std::nullptr_t 而不是类?

    • 1 个回答
  • Marko Smith

    您可以使用花括号初始化列表作为(默认)模板参数吗?

    • 2 个回答
  • Marko Smith

    为什么列表推导式在内部创建一个函数?

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Marko Smith

    java.lang.NoSuchMethodError: 'void org.openqa.selenium.remote.http.ClientConfig.<init>(java.net.URI, java.time.Duration, java.time.Duratio

    • 3 个回答
  • Marko Smith

    为什么 'char -> int' 是提升,而 'char -> Short' 是转换(但不是提升)?

    • 4 个回答
  • Marko Smith

    为什么库中不调用全局变量的构造函数?

    • 1 个回答
  • Marko Smith

    std::common_reference_with 在元组上的行为不一致。哪个是对的?

    • 1 个回答
  • Marko Smith

    C++17 中 std::byte 只能按位运算?

    • 1 个回答
  • Martin Hope
    fbrereto 为什么在传递 {} 时重载解析更喜欢 std::nullptr_t 而不是类? 2023-12-21 00:31:04 +0800 CST
  • Martin Hope
    比尔盖子 您可以使用花括号初始化列表作为(默认)模板参数吗? 2023-12-17 10:02:06 +0800 CST
  • Martin Hope
    Amir reza Riahi 为什么列表推导式在内部创建一个函数? 2023-11-16 20:53:19 +0800 CST
  • Martin Hope
    Michael A fmt 格式 %H:%M:%S 不带小数 2023-11-11 01:13:05 +0800 CST
  • Martin Hope
    God I Hate Python C++20 的 std::views::filter 未正确过滤视图 2023-08-27 18:40:35 +0800 CST
  • Martin Hope
    LiDa Cute 为什么 'char -> int' 是提升,而 'char -> Short' 是转换(但不是提升)? 2023-08-24 20:46:59 +0800 CST
  • Martin Hope
    jabaa 为什么库中不调用全局变量的构造函数? 2023-08-18 07:15:20 +0800 CST
  • Martin Hope
    Panagiotis Syskakis std::common_reference_with 在元组上的行为不一致。哪个是对的? 2023-08-17 21:24:06 +0800 CST
  • Martin Hope
    Alex Guteniev 为什么编译器在这里错过矢量化? 2023-08-17 18:58:07 +0800 CST
  • Martin Hope
    wimalopaan C++17 中 std::byte 只能按位运算? 2023-08-17 17:13:58 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve