• 首页 首页 icon
  • 工具库 工具库 icon
    • IP查询 IP查询 icon
  • 内容库 内容库 icon
    • 快讯库 快讯库 icon
    • 精品库 精品库 icon
    • 问答库 问答库 icon
  • 更多 更多 icon
    • 服务条款 服务条款 icon

11个让你成为更好的 Typescript 程序员

武飞扬头像
产品大道
帮助1

学习 Typescript 通常是一次重新发现之旅。您的最初印象可能非常具有欺骗性:这不就是一种注释 Javascript 的方式,所以编译器可以帮助我找到潜在的错误吗?

学新通

通过 r/mevlix@reddit

虽然这句话通常是正确的,但随着您继续前进,您会发现该语言最不可思议的力量在于组合、推断和操纵类型。

本文将总结几个技巧,帮助您充分发挥该语言的潜力。

#1 思考{Set}

类型对于程序员来说是一个日常概念,但要简洁地定义它却出奇地困难。我发现将Set用作概念模型会很有帮助。

例如,新学习者发现 Typescript 的组合类型的方式违反直觉。举个很简单的例子:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Measure</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span> <span style="color:var(--syntax-text-color)">};</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Style</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">color</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span> <span style="color:var(--syntax-text-color)">};</span>
  3.  
     
  4.  
    <span style="color:var(--syntax-comment-color)">// typed { radius: number; color: string }</span>
  5.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">Measure</span> <span style="color:var(--syntax-error-color)">&</span> <span style="color:var(--syntax-name-color)">Style</span><span style="color:var(--syntax-text-color)">;</span>
  6.  
    </code></span></span>
 

如果您&在逻辑 AND 的意义上解释运算符,您可能希望Circle它是一个虚拟类型,因为它是两种类型的结合,没有任何重叠字段。这不是打字稿的工作方式。相反,在Set中思考更容易推断出正确的行为:

  • 每种类型都是一值。
  • 有些集合是无限的:字符串、数字;一些有限的:布尔值,未定义的,......
  • unknown通用集(包括所有值),而never空集(包括无值)。
  • 类型Measure是所有对象的集合,包含一个名为 的数字字段radius。与 相同Style
  • &运算符创建一个IntersectionMeasure & Style表示包含radius和字段的一组对象color,这实际上是一个较小的 Set,但具有更常用的字段。
  • 类似地,|运算符创建一个Union:一个更大的 Set 但可能具有更少的常用字段(如果组合了两个对象类型)。

Set还有助于理解可赋值性:仅当值的类型是目标类型的子集时才允许赋值:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ShapeKind</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">foo</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">getSomeString</span><span style="color:var(--syntax-text-color)">();</span>
  3.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">ShapeKind</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
     
  5.  
    <span style="color:var(--syntax-comment-color)">// disallowed because string is not subset of ShapeKind</span>
  6.  
    <span style="color:var(--syntax-name-color)">shape</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">foo</span><span style="color:var(--syntax-text-color)">;</span>
  7.  
     
  8.  
    <span style="color:var(--syntax-comment-color)">// allowed because ShapeKind is subset of string</span>
  9.  
    <span style="color:var(--syntax-name-color)">foo</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">;</span>
  10.  
    </code></span></span>
 

以下文章对 Set 中的思维方式进行了出色的详尽介绍。

TypeScript 和集合论 | 伊万·奥韦耶罗

集合论如何帮助理解 TypeScript 中的类型可分配性和解析度?

学新通

ivov.dev

#2 理解声明的类型和缩小的类型

一个极其强大的打字稿功能是基于控制流的自动类型缩小。这意味着变量在代码位置的任何特定点都有两种类型与之关联:声明类型和缩小类型。

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">foo</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">x</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">if</span> <span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-declaration-color)">typeof</span> <span style="color:var(--syntax-name-color)">x</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">string</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  3.  
    <span style="color:var(--syntax-comment-color)">// x's type is narrowed to string, so .length is valid</span>
  4.  
    <span style="color:var(--syntax-name-color)">console</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">log</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">x</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">length</span><span style="color:var(--syntax-text-color)">);</span>
  5.  
     
  6.  
    <span style="color:var(--syntax-comment-color)">// assignment respects declaration type, not narrowed type</span>
  7.  
    <span style="color:var(--syntax-name-color)">x</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-literal-color)">1</span><span style="color:var(--syntax-text-color)">;</span>
  8.  
    <span style="color:var(--syntax-name-color)">console</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">log</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">x</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">length</span><span style="color:var(--syntax-text-color)">);</span> <span style="color:var(--syntax-comment-color)">// disallowed because x is now number</span>
  9.  
    <span style="color:var(--syntax-text-color)">}</span> <span style="color:var(--syntax-declaration-color)">else</span> <span style="color:var(--syntax-text-color)">{</span>
  10.  
    <span style="color:var(--syntax-text-color)">...</span>
  11.  
    <span style="color:var(--syntax-text-color)">}</span>
  12.  
    <span style="color:var(--syntax-text-color)">}</span>
  13.  
    </code></span></span>
 

#3 使用有区别的联合而不是可选字段

在定义一组像 Shape 这样的多态类型时,很容易从以下开始:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Shape</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    <span style="color:var(--syntax-text-color)">radius</span><span style="color:var(--syntax-text-color)">?:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
    <span style="color:var(--syntax-text-color)">width</span><span style="color:var(--syntax-text-color)">?:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  5.  
    <span style="color:var(--syntax-text-color)">height</span><span style="color:var(--syntax-text-color)">?:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  6.  
    <span style="color:var(--syntax-text-color)">}</span>
  7.  
     
  8.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">getArea</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  9.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">?</span>
  10.  
    <span style="color:var(--syntax-text-color)">Math</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">PI</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-error-color)">!</span> <span style="color:var(--syntax-error-color)">**</span> <span style="color:var(--syntax-literal-color)">2</span>
  11.  
    <span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">width</span><span style="color:var(--syntax-error-color)">!</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">height</span><span style="color:var(--syntax-error-color)">!</span><span style="color:var(--syntax-text-color)">;</span>
  12.  
    <span style="color:var(--syntax-text-color)">}</span>
  13.  
    </code></span></span>
 

非空断言(访问radiuswidth和字段时)是必需的,因为与其他字段height之间没有建立关系。kind相反,discriminated union 是一个更好的解决方案:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span> <span style="color:var(--syntax-text-color)">};</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Rect</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">width</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">height</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span> <span style="color:var(--syntax-text-color)">};</span>
  3.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Shape</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-name-color)">Rect</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
     
  5.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">getArea</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  6.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">?</span>
  7.  
    <span style="color:var(--syntax-text-color)">Math</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">PI</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">radius</span> <span style="color:var(--syntax-error-color)">**</span> <span style="color:var(--syntax-literal-color)">2</span>
  8.  
    <span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">width</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">height</span><span style="color:var(--syntax-text-color)">;</span>
  9.  
    <span style="color:var(--syntax-text-color)">}</span>
  10.  
    </code></span></span>
 

类型缩小消除了强制转换的需要。

#4 使用类型谓词来避免类型断言

如果你以正确的方式使用打字稿,你应该很少会发现自己使用显式类型断言(比如value as SomeType);但是,有时您仍然会感到冲动,例如:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span> <span style="color:var(--syntax-text-color)">};</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Rect</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">width</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">height</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span> <span style="color:var(--syntax-text-color)">};</span>
  3.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Shape</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-name-color)">Rect</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
     
  5.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">isCircle</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  6.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  7.  
    <span style="color:var(--syntax-text-color)">}</span>
  8.  
     
  9.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">isRect</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  10.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  11.  
    <span style="color:var(--syntax-text-color)">}</span>
  12.  
     
  13.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">myShapes</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">[]</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">getShapes</span><span style="color:var(--syntax-text-color)">();</span>
  14.  
     
  15.  
    <span style="color:var(--syntax-comment-color)">// error because typescript doesn't know the filtering</span>
  16.  
    <span style="color:var(--syntax-comment-color)">// narrows typing</span>
  17.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">circles</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Circle</span><span style="color:var(--syntax-text-color)">[]</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">myShapes</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">filter</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">isCircle</span><span style="color:var(--syntax-text-color)">);</span>
  18.  
     
  19.  
    <span style="color:var(--syntax-comment-color)">// you may be inclined to add an assertion:</span>
  20.  
    <span style="color:var(--syntax-comment-color)">// const circles = myShapes.filter(isCircle) as Circle[];</span>
  21.  
    </code></span></span>
学新通
 

一个更优雅的解决方案是改为更改isCircleisRect返回类型谓词,这样它们可以帮助 Typescript 在调用后进一步缩小类型范围filter

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">isCircle</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">):</span> <span style="color:var(--syntax-name-color)">shape</span> <span style="color:var(--syntax-declaration-color)">is</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    <span style="color:var(--syntax-text-color)">}</span>
  4.  
     
  5.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">isRect</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">):</span> <span style="color:var(--syntax-name-color)">shape</span> <span style="color:var(--syntax-declaration-color)">is</span> <span style="color:var(--syntax-name-color)">Rect</span> <span style="color:var(--syntax-text-color)">{</span>
  6.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span> <span style="color:var(--syntax-error-color)">===</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  7.  
    <span style="color:var(--syntax-text-color)">}</span>
  8.  
     
  9.  
    <span style="color:var(--syntax-text-color)">...</span>
  10.  
    <span style="color:var(--syntax-comment-color)">// now you get Circle[] type inferred correctly</span>
  11.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">circles</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">myShapes</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">filter</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">isCircle</span><span style="color:var(--syntax-text-color)">);</span>
  12.  
    </code></span></span>
 

#5 控制联合类型的分布方式

类型推断是 Typescript 的本能;大多数时候,它会默默地为你工作。但是,您可能需要对含糊不清的细微情况进行干预。分布式条件类型就是其中一种情况。

假设我们有一个ToArray辅助类型,如果输入类型还不是一个,它会返回一个数组类型:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ToArray</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">T</span> <span style="color:var(--syntax-declaration-color)">extends</span> <span style="color:var(--syntax-text-color)">Array</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">unknown</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-text-color)">?</span> <span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-text-color)">[];</span>
  2.  
    </code></span></span>
 

对于以下类型,您认为应该推断出什么?

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Foo</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">ToArray</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-error-color)">|</span><span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  2.  
    </code></span></span>
 

答案是string[] | number[]。但这是模棱两可的。为什么不(string | number)[]呢?

默认情况下,当 typescript 遇到string | number泛型参数(T此处)的联合类型(此处)时,它会分配到每个成分中,这就是您得到string[] | number[]. 这种行为可以通过使用特殊语法并包装T在一对 中来改变[],例如:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ToArray</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-text-color)">]</span> <span style="color:var(--syntax-declaration-color)">extends</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-text-color)">Array</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">unknown</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">]</span> <span style="color:var(--syntax-text-color)">?</span> <span style="color:var(--syntax-name-color)">T</span> <span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-text-color)">[];</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Foo</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">ToArray</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-declaration-color)">string</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    </code></span></span>
 

NowFoo被推断为 type (string | number)[]

#6 在编译时使用详尽检查来捕获未处理的情况

当对枚举进行 switch-casing 时,一个好习惯是主动为不期望的情况犯错,而不是像在其他编程语言中那样默默地忽略它们:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">getArea</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">switch</span> <span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  3.  
    <span style="color:var(--syntax-declaration-color)">case</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">:</span>
  4.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-text-color)">Math</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">PI</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">radius</span> <span style="color:var(--syntax-error-color)">**</span> <span style="color:var(--syntax-literal-color)">2</span><span style="color:var(--syntax-text-color)">;</span>
  5.  
    <span style="color:var(--syntax-declaration-color)">case</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">:</span>
  6.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">width</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">height</span><span style="color:var(--syntax-text-color)">;</span>
  7.  
    <span style="color:var(--syntax-text-color)">default</span><span style="color:var(--syntax-text-color)">:</span>
  8.  
    <span style="color:var(--syntax-declaration-color)">throw</span> <span style="color:var(--syntax-declaration-color)">new</span> <span style="color:var(--syntax-text-color)">Error</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">Unknown shape kind</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">);</span>
  9.  
    <span style="color:var(--syntax-text-color)">}</span>
  10.  
    <span style="color:var(--syntax-text-color)">}</span>
  11.  
    </code></span></span>
 

使用 Typescript,您可以让静态类型检查通过使用以下never类型更早地为您找到错误:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">getArea</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Shape</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">switch</span> <span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  3.  
    <span style="color:var(--syntax-declaration-color)">case</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">:</span>
  4.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-text-color)">Math</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">PI</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">radius</span> <span style="color:var(--syntax-error-color)">**</span> <span style="color:var(--syntax-literal-color)">2</span><span style="color:var(--syntax-text-color)">;</span>
  5.  
    <span style="color:var(--syntax-declaration-color)">case</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">:</span>
  6.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">width</span> <span style="color:var(--syntax-error-color)">*</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">height</span><span style="color:var(--syntax-text-color)">;</span>
  7.  
    <span style="color:var(--syntax-text-color)">default</span><span style="color:var(--syntax-text-color)">:</span>
  8.  
    <span style="color:var(--syntax-comment-color)">// you'll get a type-checking error below </span>
  9.  
    <span style="color:var(--syntax-comment-color)">// if any shape.kind is not handled above</span>
  10.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">_exhaustiveCheck</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">never</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">;</span>
  11.  
    <span style="color:var(--syntax-declaration-color)">throw</span> <span style="color:var(--syntax-declaration-color)">new</span> <span style="color:var(--syntax-text-color)">Error</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">Unknown shape kind</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">);</span>
  12.  
    <span style="color:var(--syntax-text-color)">}</span>
  13.  
    <span style="color:var(--syntax-text-color)">}</span>
  14.  
    </code></span></span>
 

有了这个,就不可能getArea在添加新的形状种类时忘记更新函数。

该技术背后的基本原理是该never类型不能分配任何东西,除了nevershape.kind如果case 语句穷尽了所有候选者,则唯一可能到达的类型default是 never;但是,如果任何候选人未被涵盖,它将泄漏到default分支并导致无效分配。

type#7更 喜欢interface

在打字稿中,当用于键入对象时type,它们是非常相似的结构。尽管可能存在争议,但我的建议是在大多数情况下interface始终使用,并且仅在满足以下任一条件时才使用:typeinterface

  • 您想要利用 的“合并”功能interface

  • 您有涉及类/接口层次结构的 OO 样式代码。

否则,始终使用更通用的type构造会产生更一致的代码。

#8 在适当的时候优先使用元组而不是数组

对象类型是输入结构化数据的常用方式,但有时您可能需要更简洁的表示方式,而是使用简单的数组。例如,我们Circle可以这样定义:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-declaration-color)">string</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">)[];</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-literal-color)">1.0</span><span style="color:var(--syntax-text-color)">];</span> <span style="color:var(--syntax-comment-color)">// [kind, radius]</span>
  3.  
    </code></span></span>
 

但是这种输入不必要地松散,您可以通过创建类似['circle', '1.0']. 我们可以通过使用 Tuple 来使其更严格:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">];</span>
  2.  
     
  3.  
    <span style="color:var(--syntax-comment-color)">// you'll get an error below</span>
  4.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">Circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">1.0</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">];</span>
  5.  
    </code></span></span>
 

元组用法的一个很好的例子是 React 的useState.

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">setName</span><span style="color:var(--syntax-text-color)">]</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">useState</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-string-color)">''</span><span style="color:var(--syntax-text-color)">);</span>
  2.  
    </code></span></span>
 

它既紧凑又类型安全。

#9 控制推断类型的一般性或具体性

Typescript 在进行类型推断时使用合理的默认行为,旨在简化常见情况下的代码编写(因此类型不需要显式注释)。有几种方法可以调整它的行为。

  • 用于const缩小到最具体的类型
  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">foo</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">foo</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">};</span> <span style="color:var(--syntax-comment-color)">// typed: { name: string }</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">Bar</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">bar</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">}</span> <span style="color:var(--syntax-declaration-color)">as</span> <span style="color:var(--syntax-declaration-color)">const</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-comment-color)">// typed: { name: 'bar' }</span>
  3.  
     
  4.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">a</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-literal-color)">1</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-literal-color)">2</span><span style="color:var(--syntax-text-color)">];</span> <span style="color:var(--syntax-comment-color)">// typed: number[]</span>
  5.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">b</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-literal-color)">1</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-literal-color)">2</span><span style="color:var(--syntax-text-color)">]</span> <span style="color:var(--syntax-declaration-color)">as</span> <span style="color:var(--syntax-declaration-color)">const</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-comment-color)">// typed: [1, 2]</span>
  6.  
     
  7.  
    <span style="color:var(--syntax-comment-color)">// typed { kind: 'circle; radius: number }</span>
  8.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-declaration-color)">as</span> <span style="color:var(--syntax-declaration-color)">const</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-literal-color)">1.0</span> <span style="color:var(--syntax-text-color)">};</span>
  9.  
     
  10.  
    <span style="color:var(--syntax-comment-color)">// the following won't work if circle wasn't initialized</span>
  11.  
    <span style="color:var(--syntax-comment-color)">// with the const keyword</span>
  12.  
    <span style="color:var(--syntax-declaration-color)">let</span> <span style="color:var(--syntax-name-color)">shape</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-text-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">rect</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">}</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">;</span>
  13.  
    </code></span></span>
 
  • 用于satisfies检查类型而不影响推断类型

考虑以下示例:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">NamedCircle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    <span style="color:var(--syntax-text-color)">name</span><span style="color:var(--syntax-text-color)">?:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
    <span style="color:var(--syntax-text-color)">};</span>
  5.  
     
  6.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">NamedCircle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-literal-color)">1.0</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">yeah</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">};</span>
  7.  
     
  8.  
    <span style="color:var(--syntax-comment-color)">// error because circle.name can be undefined</span>
  9.  
    <span style="color:var(--syntax-name-color)">console</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">log</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">length</span><span style="color:var(--syntax-text-color)">);</span>
  10.  
    </code></span></span>
 

我们得到一个错误,因为根据circle的声明类型NamedCirclename字段确实可以是未定义的,即使变量初始值设定项提供了一个字符串值。当然,我们可以删除类型注释,但我们将放弃对对象: NamedCircle有效性的类型检查。circle相当进退两难。

幸运的是,Typescript 4.9 引入了一个新satisfies关键字,它允许您在不改变推断类型的情况下检查类型:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">NamedCircle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    <span style="color:var(--syntax-text-color)">name</span><span style="color:var(--syntax-text-color)">?:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
    <span style="color:var(--syntax-text-color)">};</span>
  5.  
     
  6.  
    <span style="color:var(--syntax-comment-color)">// error because radius violates NamedCircle</span>
  7.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">wrongCircle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">1.0</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">ha</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">}</span>
  8.  
    <span style="color:var(--syntax-name-color)">satisfies</span> <span style="color:var(--syntax-name-color)">NamedCircle</span><span style="color:var(--syntax-text-color)">;</span>
  9.  
     
  10.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">circle</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-literal-color)">1.0</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">yeah</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-text-color)">}</span>
  11.  
    <span style="color:var(--syntax-name-color)">satisfies</span> <span style="color:var(--syntax-name-color)">NamedCircle</span><span style="color:var(--syntax-text-color)">;</span>
  12.  
     
  13.  
    <span style="color:var(--syntax-comment-color)">// circle.name can't be undefined now</span>
  14.  
    <span style="color:var(--syntax-name-color)">console</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">log</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">name</span><span style="color:var(--syntax-text-color)">.</span><span style="color:var(--syntax-name-color)">length</span><span style="color:var(--syntax-text-color)">);</span>
  15.  
    </code></span></span>
学新通
 

修改后的版本享有两个好处:保证对象文字符合NamedCircle类型,并且推断类型具有不可为空的name字段。

#10infer用于创建额外的泛型类型参数

在设计实用函数和类型时,您经常会觉得需要使用从给定类型参数中提取的类型。在这种情况下,infer关键字就派上用场了。它可以帮助您即时推断出新的类型参数。这里有两个简单的例子:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-comment-color)">// gets the unwrapped type out of a Promise;</span>
  2.  
    <span style="color:var(--syntax-comment-color)">// idempotent if T is not Promise</span>
  3.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ResolvedPromise</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">T</span> <span style="color:var(--syntax-declaration-color)">extends</span> <span style="color:var(--syntax-text-color)">Promise</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">infer</span> <span style="color:var(--syntax-name-color)">U</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-text-color)">?</span> <span style="color:var(--syntax-name-color)">U</span> <span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">t</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">ResolvedPromise</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-text-color)">Promise</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-error-color)">>></span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-comment-color)">// t: string</span>
  5.  
     
  6.  
    <span style="color:var(--syntax-comment-color)">// gets the flattened type of array T;</span>
  7.  
    <span style="color:var(--syntax-comment-color)">// idempotent if T is not array</span>
  8.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Flatten</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">T</span> <span style="color:var(--syntax-declaration-color)">extends</span> <span style="color:var(--syntax-text-color)">Array</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">infer</span> <span style="color:var(--syntax-name-color)">E</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-text-color)">?</span> <span style="color:var(--syntax-name-color)">Flatten</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">E</span><span style="color:var(--syntax-error-color)">></span> <span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">T</span><span style="color:var(--syntax-text-color)">;</span>
  9.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">e</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">Flatten</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">[][]</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-comment-color)">// e: number</span>
  10.  
    </code></span></span>
 

infer关键字的工作原理可以T extends Promise<infer U>理解为:假设 T 与某些实例化的通用 Promise 类型兼容,即兴创作一个类型参数 U 使其工作。因此,如果T被实例化为Promise<string>,则 的解U将是string

#11通过在类型操作上发挥创意来 保持DRY

Typescript 提供了强大的类型操作语法和一组非常有用的实用程序,可帮助您将代码重复减少到最低限度。这里只是一些特别的例子:

  • 而不是复制字段声明:
  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">User</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-name-color)">age</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    <span style="color:var(--syntax-text-color)">gender</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
    <span style="color:var(--syntax-text-color)">country</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span>
  5.  
    <span style="color:var(--syntax-text-color)">city</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span>
  6.  
    <span style="color:var(--syntax-text-color)">};</span>
  7.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Demographic</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">age</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">number</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">gender</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">};</span>
  8.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Geo</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">country</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">city</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">};</span>
  9.  
    </code></span></span>
 

,使用Pick实用程序提取新类型:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">User</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-name-color)">age</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span><span style="color:var(--syntax-text-color)">;</span>
  3.  
    <span style="color:var(--syntax-text-color)">gender</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span>
  4.  
    <span style="color:var(--syntax-text-color)">country</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span><span style="color:var(--syntax-text-color)">;</span>
  5.  
    <span style="color:var(--syntax-text-color)">city</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">string</span>
  6.  
    <span style="color:var(--syntax-text-color)">};</span>
  7.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Demographic</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">Pick</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">User</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">age</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-error-color)">|</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">gender</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  8.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Geo</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">Pick</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">User</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">country</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-error-color)">|</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">city</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  9.  
    </code></span></span>
 
  • 而不是复制函数的返回类型
  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">createCircle</span><span style="color:var(--syntax-text-color)">()</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-text-color)">{</span>
  3.  
    <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-declaration-color)">as</span> <span style="color:var(--syntax-declaration-color)">const</span><span style="color:var(--syntax-text-color)">,</span>
  4.  
    <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-literal-color)">1.0</span>
  5.  
    <span style="color:var(--syntax-text-color)">}</span>
  6.  
    <span style="color:var(--syntax-text-color)">}</span>
  7.  
     
  8.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">transformCircle</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-text-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span> <span style="color:var(--syntax-text-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">number</span> <span style="color:var(--syntax-text-color)">})</span> <span style="color:var(--syntax-text-color)">{</span>
  9.  
    <span style="color:var(--syntax-text-color)">...</span>
  10.  
    <span style="color:var(--syntax-text-color)">}</span>
  11.  
     
  12.  
    <span style="color:var(--syntax-name-color)">transformCircle</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">createCircle</span><span style="color:var(--syntax-text-color)">());</span>
  13.  
    </code></span></span>
 

,用于ReturnType<T>提取它:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">createCircle</span><span style="color:var(--syntax-text-color)">()</span> <span style="color:var(--syntax-text-color)">{</span>
  2.  
    <span style="color:var(--syntax-declaration-color)">return</span> <span style="color:var(--syntax-text-color)">{</span>
  3.  
    <span style="color:var(--syntax-name-color)">kind</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">circle</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-declaration-color)">as</span> <span style="color:var(--syntax-declaration-color)">const</span><span style="color:var(--syntax-text-color)">,</span>
  4.  
    <span style="color:var(--syntax-name-color)">radius</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-literal-color)">1.0</span>
  5.  
    <span style="color:var(--syntax-text-color)">}</span>
  6.  
    <span style="color:var(--syntax-text-color)">}</span>
  7.  
     
  8.  
    <span style="color:var(--syntax-declaration-color)">function</span> <span style="color:var(--syntax-name-color)">transformCircle</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">circle</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">ReturnType</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-declaration-color)">typeof</span> <span style="color:var(--syntax-name-color)">createCircle</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">)</span> <span style="color:var(--syntax-text-color)">{</span>
  9.  
    <span style="color:var(--syntax-text-color)">...</span>
  10.  
    <span style="color:var(--syntax-text-color)">}</span>
  11.  
     
  12.  
    <span style="color:var(--syntax-name-color)">transformCircle</span><span style="color:var(--syntax-text-color)">(</span><span style="color:var(--syntax-name-color)">createCircle</span><span style="color:var(--syntax-text-color)">());</span>
  13.  
    </code></span></span>
 
  • 而不是并行同步两种类型的形状(此处为 typeof config 和 Factory):
  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ContentTypes</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">news</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">blog</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">video</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  2.  
     
  3.  
    <span style="color:var(--syntax-comment-color)">// config for indicating what content types are enabled</span>
  4.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">config</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">news</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">true</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">blog</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">true</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">video</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">false</span> <span style="color:var(--syntax-text-color)">}</span>
  5.  
    <span style="color:var(--syntax-name-color)">satisfies</span> <span style="color:var(--syntax-text-color)">Record</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">ContentTypes</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">boolean</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  6.  
     
  7.  
    <span style="color:var(--syntax-comment-color)">// factory for creating contents</span>
  8.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Factory</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  9.  
    <span style="color:var(--syntax-name-color)">createNews</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-text-color)">()</span> <span style="color:var(--syntax-error-color)">=></span> <span style="color:var(--syntax-name-color)">Content</span><span style="color:var(--syntax-text-color)">;</span>
  10.  
    <span style="color:var(--syntax-text-color)">createBlog</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-text-color)">()</span> <span style="color:var(--syntax-error-color)">=></span> <span style="color:var(--syntax-name-color)">Content</span><span style="color:var(--syntax-text-color)">;</span>
  11.  
    <span style="color:var(--syntax-text-color)">};</span>
  12.  
    </code></span></span>
 

,使用Mapped TypeTemplate Literal Type根据配置的形状自动推断出正确的工厂类型:

  1.  
    <span style="color:var(--syntax-text-color)"><span style="color:var(--syntax-text-color)"><code><span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ContentTypes</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">news</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">blog</span><span style="color:var(--syntax-string-color)">'</span> <span style="color:var(--syntax-error-color)">|</span> <span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-string-color)">video</span><span style="color:var(--syntax-string-color)">'</span><span style="color:var(--syntax-text-color)">;</span>
  2.  
     
  3.  
    <span style="color:var(--syntax-comment-color)">// generic factory type with a inferred list of methods</span>
  4.  
    <span style="color:var(--syntax-comment-color)">// based on the shape of the given Config</span>
  5.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">ContentFactory</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">Config</span> <span style="color:var(--syntax-declaration-color)">extends</span> <span style="color:var(--syntax-text-color)">Record</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">ContentTypes</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">boolean</span><span style="color:var(--syntax-error-color)">>></span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span>
  6.  
    <span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-name-color)">k</span> <span style="color:var(--syntax-declaration-color)">in</span> <span style="color:var(--syntax-declaration-color)">string</span> <span style="color:var(--syntax-error-color)">&</span> <span style="color:var(--syntax-declaration-color)">keyof</span> <span style="color:var(--syntax-name-color)">Config</span> <span style="color:var(--syntax-declaration-color)">as</span> <span style="color:var(--syntax-name-color)">Config</span><span style="color:var(--syntax-text-color)">[</span><span style="color:var(--syntax-name-color)">k</span><span style="color:var(--syntax-text-color)">]</span> <span style="color:var(--syntax-declaration-color)">extends</span> <span style="color:var(--syntax-declaration-color)">true</span>
  7.  
    <span style="color:var(--syntax-text-color)">?</span> <span style="color:var(--syntax-string-color)">`create</span><span style="color:var(--syntax-text-color)">${</span><span style="color:var(--syntax-name-color)">Capitalize</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">k</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">}</span><span style="color:var(--syntax-string-color)">`</span>
  8.  
    <span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-name-color)">never</span><span style="color:var(--syntax-text-color)">]:</span> <span style="color:var(--syntax-text-color)">()</span> <span style="color:var(--syntax-error-color)">=></span> <span style="color:var(--syntax-name-color)">Content</span><span style="color:var(--syntax-text-color)">;</span>
  9.  
    <span style="color:var(--syntax-text-color)">};</span>
  10.  
     
  11.  
    <span style="color:var(--syntax-comment-color)">// config for indicating what content types are enabled</span>
  12.  
    <span style="color:var(--syntax-declaration-color)">const</span> <span style="color:var(--syntax-name-color)">config</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-text-color)">{</span> <span style="color:var(--syntax-name-color)">news</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">true</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">blog</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">true</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">video</span><span style="color:var(--syntax-text-color)">:</span> <span style="color:var(--syntax-declaration-color)">false</span> <span style="color:var(--syntax-text-color)">}</span>
  13.  
    <span style="color:var(--syntax-name-color)">satisfies</span> <span style="color:var(--syntax-text-color)">Record</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-name-color)">ContentTypes</span><span style="color:var(--syntax-text-color)">,</span> <span style="color:var(--syntax-name-color)">boolean</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  14.  
     
  15.  
    <span style="color:var(--syntax-declaration-color)">type</span> <span style="color:var(--syntax-name-color)">Factory</span> <span style="color:var(--syntax-error-color)">=</span> <span style="color:var(--syntax-name-color)">ContentFactory</span><span style="color:var(--syntax-error-color)"><</span><span style="color:var(--syntax-declaration-color)">typeof</span> <span style="color:var(--syntax-name-color)">config</span><span style="color:var(--syntax-error-color)">></span><span style="color:var(--syntax-text-color)">;</span>
  16.  
    <span style="color:var(--syntax-comment-color)">// Factory: {</span>
  17.  
    <span style="color:var(--syntax-comment-color)">// createNews: () => Content;</span>
  18.  
    <span style="color:var(--syntax-comment-color)">// createBlog: () => Content; </span>
  19.  
    <span style="color:var(--syntax-comment-color)">// }</span>
  20.  
    </code></span></span>
学新通
 

发挥您的想象力,您会发现无限的探索潜力。

包起来

这篇文章涵盖了 Typescript 语言中的一组相对高级的主题。实际上,您可能会发现直接应用它们并不常见;然而,此类技术被专门为 Typescript 设计的库大量使用:如PrismatRPC。了解这些技巧可以帮助您更好地了解这些工具如何在幕后发挥其魔力。


PS 我们正在构建ZenStack — 一个用于使用 Next.js Typescript 构建安全 CRUD 应用程序的工具包。我们的目标是让您节省编写样板代码的时间,并专注于构建重要的东西——用户体验。

这篇好文章是转载于:学新通技术网

  • 版权申明: 本站部分内容来自互联网,仅供学习及演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,请提供相关证据及您的身份证明,我们将在收到邮件后48小时内删除。
  • 本站站名: 学新通技术网
  • 本文地址: /boutique/detail/tanhgghagb
系列文章
更多 icon
同类精品
更多 icon
继续加载