写了很久TypeScript,才发现这些写法不对

更新日期: 2026-03-05 阅读: 34 标签: TypeScript

很多人用TypeScript用得不爽,不是不会语法

是经常遇到这种情况:

  • 代码能跑,但心里没底

  • 类型写了,但还是出Bug

  • 看别人代码,类型一大坨,看不懂

问题不在TypeScript本身,是类型没贴着业务写。


1. 接口返回不稳定,别直接用any

后端接口返回用户信息,但有时候字段不全,甚至返回null。

很多人第一反应是:

function formatUser(user: any) {
  return user.name.toUpperCase();
}

这段代码的问题是:TypeScript已经帮不上任何忙了。

换个写法:

function formatUser(user: unknown) {
  if (
    typeof user === 'object' &&
    user !== null &&
    'name' in user
  ) {
    const name = (user as { name: string }).name;
    return name.toUpperCase();
  }

  return '--';
}

不复杂,但有两个好处:接口异常时不会直接炸,一眼就能看出哪里做了兜底。


2. 状态码要限制范围

接口返回状态码:1成功,2失败,3处理中。

很多人这样写:

function getStatusText(status: number) {
  if (status === 1) return '成功';
  if (status === 2) return '失败';
  if (status === 3) return '处理中';
  return '未知';
}

问题是传99,TypeScript也不会拦。

改成联合类型:

type Status = 1 | 2 | 3;

function getStatusText(status: Status) {
  const map: Record<Status, string> = {
    1: '成功',
    2: '失败',
    3: '处理中'
  };

  return map[status];
}

现在调用getStatusText(4)会直接报错。


3. 配置和枚举要加as const

写了一个权限配置:

const ROLE = {
  ADMIN: 'admin',
  USER: 'user'
};

想在别的地方用'admin' | 'user'这个类型,但TypeScript只会给string。

正确写法:

const ROLE = {
  ADMIN: 'admin',
  USER: 'user'
} as const;

type Role = typeof ROLE[keyof typeof ROLE];
// 'admin' | 'user'

这个在函数参数里特别好用。


4. 列表映射用Record比interface清楚

接口返回的是一个对象列表:

{
  "1001": { "name": "Tom", "age": 18 },
  "1002": { "name": "Jerry", "age": 20 }
}

推荐写法:

type User = {
  name: string;
  age: number;
};

type UserMap = Record<string, User>;

一眼就知道:key是string,value是User。


5. 表单编辑页用Partial

新增用户需要全部字段,编辑用户只改一部分。

type User = {
  id: number;
  name: string;
  age: number;
};

// 编辑接口参数
type UpdateUserParams = Partial<User>;

现在这样写是合法的:

updateUser({
  id: 1,
  name: 'Tom'
});


6. 不同页面用不同字段用Pick

列表页只展示name和id,详情页展示全部。

type User = {
  id: number;
  name: string;
  age: number;
  address: string;
};

// 列表类型
type UserListItem = Pick<User, 'id' | 'name'>;

少写一套类型,也不会不一致。


7. 返回值不确定时让TypeScript自己推断

接口可能返回用户,也可能返回null。

function getUser() {
  if (Math.random() > 0.5) {
    return { name: 'Tom' };
  }
  return null;
}

TypeScript推断结果是:{ name: string } | null。

使用时:

const user = getUser();

if (user) {
  console.log(user.name);
}

不用自己手写联合类型,反而更安全。


8. 少写断言,多写判断

接口字段类型不确定:

function handle(value: unknown) {
  // 不推荐
  // (value as string).toUpperCase();
}

推荐写法:

function isString(value: unknown): value is string {
  return typeof value === 'string';
}

function handle(value: unknown) {
  if (isString(value)) {
    return value.toUpperCase();
  }
}


9. 枚举不是不能用,但普通项目没必要

const STATUS = {
  SUCCESS: 1,
  FAIL: 2
} as const;

type Status = typeof STATUS[keyof typeof STATUS];

调接口、看日志都更直观,值就是数字,不用再查对应关系。


最后

TypeScript真正难的,不是语法,是知道什么时候该严,什么时候该放。

如果你现在写TypeScript还有点别扭,其实很正常。说明你已经开始关心代码质量了。

本文内容仅供个人学习、研究或参考使用,不构成任何形式的决策建议、专业指导或法律依据。未经授权,禁止任何单位或个人以商业售卖、虚假宣传、侵权传播等非学习研究目的使用本文内容。如需分享或转载,请保留原文来源信息,不得篡改、删减内容或侵犯相关权益。感谢您的理解与支持!

链接: https://fly63.com/article/detial/13387

相关推荐

Nerv_一款类 React 前端框架,基于虚拟 DOM 技术的 JavaScript(TypeScript) 库

Nerv_是一款由京东凹凸实验室打造的类 React 前端框架,基于虚拟 DOM 技术的 JavaScript(TypeScript) 库。它基于React标准,提供了与 React 16 一致的使用方式与 API。

使用TypeScript两年后-值得吗?

差不多两年前,我在一个创业团队中开始了一个全新的项目。用到的全都是类似Microservices,docker,react,redux这些时髦的东西。我在前端技术方面积累了一些类似的经验

TypeScript最佳实践:是否使用noImplicitAny

我应该使用noImplicitAny TypeScript编译器标志吗?noImplicitAny编译器选项所做的,基本上是将TypeScript从可选类型语言转换为强制类型检验语言。这使得TypeScript离JavaScript的超集稍微远了一些,因为简单的:

为什么要学习Typescript 语言呢?Typescript 开发环境安装

TypeScript是一种由微软开发的自由和开源的编程语言。它是JavaScript的一个超集,TypeScript是JavaScript类型的超集,它可以编译成纯JavaScript。TypeScript可以在任何浏览器、任何计算机和任何操作系统上运行,并且是开源的。

5分钟了解TypeScript

有两种方式安装TypeScript,如何创建第一个TypeScript文件,在TypeScript中,可以使用interface来描述一个对象有firstName和lastName两个属性,TypeScript支持JavaScript的新功能,其中很重要的一个功能就是基于类的面向对象编程

Typescript中以变量方式传递类

最近尝试用TypeScript写一个工具库,需要实现这样一个场景:声明一个抽象类Parent,声明一组子类ChildA、ChildB继承这个Parent,实现它的抽象方法

TypeScript_TS系列之高级类型

交叉类型:将多个类型合并为一个类型、联合类型:表示取值可以为多种类型中的一种、混合类型:一个例子就是,一个对象可以同时做为函数和对象使用,并带有额外的属性、类型断言:可以用来手动指定一个值的类型

用TypeScript弥补Elm和JavaScript之间的差距

近些日子,我使用了新语言编程,从JavaScript,切确地说是Elm,转成TypeScript。在本文中,我将继续深挖一些我非常喜欢的TypeScript特性。

TypeScript_命名空间(namespace)

什么时候要用命名空间?如果你发现自己写的功能(函数/类/接口等...)越来越多, 你想对他们进行分组管理就可以用命名空间, 下面先用类,举例:发现namespace下还有export, export在这里用来表示哪些功能是可以外部访问的:

TypeScript功能:const断言

我发现官方的 TypeScript 文档非常有用,但是总觉得有点过于学术化并且枯燥无味。每当我发现一个新功能时,我想要知道这个功能究竟能够解决什么问题而不是长篇大论

点击更多...

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!