0

0

掌握 TypeScript 模板文字类型:提高代码安全性和表现力

花韻仙語

花韻仙語

发布时间:2024-11-20 18:42:01

|

836人浏览过

|

来源于dev.to

转载

掌握 typescript 模板文字类型:提高代码安全性和表现力

好吧,让我们进入使用模板文字类型在 typescript 中进行编译时元编程的迷人世界。这个强大的功能使我们能够创建一些非常酷的类型级魔法,使我们的代码更安全、更具表现力。

首先,模板文字类型到底是什么?它们是一种基于字符串文字操作和创建新类型的方法。这就像拥有一种适合您类型的迷你编程语言。很整洁,对吧?

让我们从一个简单的例子开始:

type greeting = `hello, ${t}!`;
type result = greeting<"world">; // "hello, world!"

在这里,我们创建了一个类型,它接受一个字符串并将其包装在问候语中。 typescript 编译器在编译时计算出结果类型。不过,这只是表面现象。

我们可以使用模板文字类型来创建更复杂的转换。例如,假设我们要创建一个将snake_case 转换为camelcase 的类型:

type snaketocamel = s extends `${infer t}_${infer u}`
  ? `${t}${capitalize>}`
  : s;

type result = snaketocamel<"hello_world_typescript">; // "helloworldtypescript"

此类型递归地转换输入字符串,将下划线后的每个部分大写。 infer 关键字在这里至关重要 - 它允许我们将字符串的一部分提取到新的类型变量中。

但是为什么停在那里呢?我们可以使用这些技术在我们的类型系统中构建整个领域特定语言(dsl)。想象一下创建一个类型安全的 sql 查询构建器:

type table = "users" | "posts" | "comments";
type column = "id" | "name" | "email" | "content";

type select = `select ${c} from ${t}`;
type where = `where ${t}`;

type query = 
  `${select} ${where}`;

type userquery = query<"users", "name" | "email", "id = 1">; 
// "select name, email from users where id = 1"

此设置确保我们仅从有效表中选择有效列,所有这些都在编译时进行检查。不会再出现因列名输入错误而导致的运行时错误!

我们可以通过实现更复杂的类型级计算来更进一步。让我们创建一个可以执行基本算术的类型:

type digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9';
type adddigits = 
  // ... (implementation details omitted for brevity)

type add = 
  // ... (implementation details omitted for brevity)

type result = add<"123", "456">; // "579"

此类型可以将两个表示为字符串的数字相加。实际的实现相当复杂,涉及很多条件类型和递归,但最终结果是纯粹的编译时魔法。

这些技术的一个实际应用是创建高级表单验证模式。我们可以定义一个类型来描述表单的形状并使用它来生成验证规则:

type form = {
  name: string;
  email: string;
  age: number;
};

type validationrule = t extends string
  ? "isstring"
  : t extends number
  ? "isnumber"
  : never;

type validationschema = {
  [k in keyof t]: validationrule;
};

type formvalidation = validationschema
; // { name: "isstring", email: "isstring", age: "isnumber" }

然后可以使用此模式生成运行时验证代码,确保我们的验证逻辑始终与我们的类型定义匹配。

模板文字类型还使我们能够创建更灵活的 api。我们可以使用它们通过适当的类型推断来实现方法链:

type chainable = {
  set: (key: k, value: v) => chainable;
  get: () => t;
};

declare function createchainable(): chainable;

const result = createchainable()
  .set("foo", 123)
  .set("bar", "hello")
  .get();

// result type: { foo: number, bar: string }

这种模式允许我们逐步构建对象,类型系统会跟踪每一步累积的属性。

佳蓝在线销售系统(创业版) 佳蓝在线销售
佳蓝在线销售系统(创业版) 佳蓝在线销售

1、对ASP内核代码进行DLL封装,从而大大提高了用户的访问速度和安全性;2、采用后台生成HTML网页的格式,使程序访问速度得到进一步的提升;3、用户可发展下级会员并在下级购买商品时获得差额利润;4、全新模板选择功能;5、后台增加磁盘绑定功能;6、后台增加库存查询功能;7、后台增加财务统计功能;8、后台面值类型批量设定;9、后台财务曲线报表显示;10、完善订单功能;11、对所有传输的字符串进行安全

下载

编译时元编程最强大的方面之一是能够基于现有类型生成新类型。我们可以使用它来创建实用程序类型,以有用的方式转换其他类型。例如,让我们创建一个类型,使对象的所有属性都是可选的,但仅限于第一级:

type shallowpartial = {
  [p in keyof t]?: t[p] extends object ? t[p] : t[p] | undefined;
};

type user = {
  name: string;
  address: {
    street: string;
    city: string;
  };
};

type partialuser = shallowpartial;
// {
//   name?: string | undefined;
//   address?: {
//     street: string;
//     city: string;
//   } | undefined;
// }

此类型使顶级属性可选,但使嵌套对象保持不变。它是 typescript 内置 partial 类型的更细致的版本。

我们还可以使用模板文字类型来创建更具表现力的错误消息。我们可以引导开发人员找到确切的问题,而不是出现神秘的类型错误:

type assertequal = t extends u
  ? u extends t
    ? true
    : `expected ${u}, but got ${t}`
  : `expected ${u}, but got ${t}`;

type test1 = assertequal<"hello", "hello">; // true
type test2 = assertequal<"hello", "world">; // "expected "world", but got "hello""

这种技术在库开发中特别有用,向用户提供清晰的反馈至关重要。

另一个有趣的应用是创建类型安全的事件发射器。我们可以使用模板文字类型来确保事件名称及其相应的负载正确匹配:

type eventmap = {
  "user:login": { userid: string };
  "item:added": { itemid: number; quantity: number };
};

type eventname = keyof eventmap;

class typedeventemitter> {
  emit(event: e, data: t[e]): void {
    // implementation details omitted
  }

  on(event: e, callback: (data: t[e]) => void): void {
    // implementation details omitted
  }
}

const emitter = new typedeventemitter();

emitter.emit("user:login", { userid: "123" }); // ok
emitter.emit("item:added", { itemid: 456, quantity: 2 }); // ok
emitter.emit("user:login", { itemid: 789 }); // type error!

此设置可确保我们始终发出并侦听具有正确负载类型的事件。

模板文字类型也可用于实现类型级状态机。这对于复杂工作流程或协议的建模非常有用:

type State = "idle" | "loading" | "success" | "error";

type Event = "fetch" | "success" | "failure" | "reset";

type Transition =
  S extends "idle"
    ? E extends "fetch"
      ? "loading"
      : S
    : S extends "loading"
    ? E extends "success"
      ? "success"
      : E extends "failure"
      ? "error"
      : S
    : S extends "success" | "error"
    ? E extends "reset"
      ? "idle"
      : S
    : never;

type Machine = {
  state: S;
  transition: (event: E) => Machine>;
};

declare function createMachine(initialState: S): Machine;

const machine = createMachine("idle")
  .transition("fetch")
  .transition("success")
  .transition("reset");

type FinalState = typeof machine.state; // "idle"

这个状态机是完全类型安全的 - 它不会允许无效的转换,并且会准确地跟踪当前状态。

总之,typescript 中使用模板文字类型的编译时元编程开辟了一个充满可能性的世界。它使我们能够创建更具表现力、类型安全和自文档化的代码。我们可以更早地捕获错误,提供更好的开发人员体验,甚至可以根据类型生成代码。虽然这些技术可能很复杂,但它们为构建强大而灵活的系统提供了强大的工具。与任何高级功能一样,明智地使用它们很重要 - 有时更简单的解决方案更易于维护。但如果使用得当,编译时元编程可以显着提高 typescript 代码的质量和可靠性。


我们的创作

一定要看看我们的创作:

投资者中心 | 智能生活 | 时代与回声 | 令人费解的谜团 | 印度教 | 精英开发 | js学校


我们在媒体上

科技考拉洞察 | 时代与回响世界 | 投资者中央媒体 | 令人费解的谜团 | 科学与时代媒介 | 现代印度教

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

675

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

319

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

345

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1084

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

355

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

673

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

566

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

409

2024.04.29

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 2.3万人学习

Git版本控制工具
Git版本控制工具

共8课时 | 1.5万人学习

Git中文开发手册
Git中文开发手册

共0课时 | 0人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号