比如我有下面这样一个类型:
type User = {
username: string;
gender: 'male' | 'female';
age: number;
bio: string;
password: string;
}
User 类型要求所有属性都必须有值,所以:
const user: User = {
username: 'awesomeuser',
};
是不可行的,会提示:
类型“{ username: string; }”缺少类型“User”中的以下属性: gender, age, bio
如何让它可行?使用 Partial 即可:
const user: Partial<User> = {
username: 'awesomeuser'
}
Partial 内置内型的作用就是让一个已定义了的类型的所有属性变得可选,具体实现如下:
/**
* Make all properties in T optional
*/
type Partial<T> = {
[P in keyof T]?: T[P];
};
假设我们有下面这样一个类型定义:
type User = {
username: string;
age?: number;
gender?: 'male' | 'female'
bio?: string;
password?: string;
}
然后从服务器后端拿到了一系列用户数据的结果:
const users: User[] = await api.users.list();
此时我们尝试去访问用户的 age 进行比较,想要取出 age > 18 的用户:
const filteredUsers = users.filter(user => user.age > 18);
此时你的 IDE 是不是提示你:对象可能为“未定义”。?为什么?因为我们定义了 age 本身就是可选的属性,未定义该属性时,它的值就是 undefined,而在 typescript 中, undefined 是不允许与 number 类型进行比较的,但是此时其实我们是能很确定 api.users.list 接口返回给我的,要么是抛出错误,要么给到我的就一定是带了 age 值的 User 类型数据,所以,我是完全可以去比较的,但是由于 TypeScript 并不知道你加载的数据是 User 还是所有属性都已经有值的特殊的 User,那怎么办?什么 Required 即可。
const users: Required<User>[] = [];
或者让 api.users.list() 的返回类型就是 Required<User>[] 即可。
如何自己实现 Required:
/**
* Make all properties in T required
*/
type Required<T> = {
[P in keyof T]-?: T[P];
};
假设我们现在在写一个系统,当用户登录之后,会共享一个 User 对象给所有组件,但是只允许他人读取其值,不允许他人修改,但是我们定义的 User 是整个系统共用的,有编辑功能的页面也在使用这个类型定义,它们是需要能修改用户数据的,那怎么办?
使用 Readonly 即可:
const user = {
username: "awesomeuser",
gender: "male",
age: 19,
bio: "Aha, insight~",
};
const sharedUser = user as Readonly<User>;
sharedUser.username = "new Name";
此时,IDE 就会告诉你无法分配到 "username" ,因为它是只读属性。
注意:TypeScript 只作静态类型校验,但是并不表示这些值真的不能被修改了,如果真的要创建一个真正从程序上都不能被修改的对象,请问怎么解决?
还是那个 User,我想定义一个 Login 类型,它专门用于登录时的类型,但是我又不想重新去写一遍 username 与 password 的类型定义(虽然在本例中,重新写一遍也很简单,但保不齐哪天就会遇到一个十分复杂的类型定义呢?),怎么办?使用 Pick 即可。
type User = {
username: string;
gender?: "male" | "female";
age?: number;
bio?: string;
password?: string;
};
type Login = Pick<User, "username" | "password">;
const login: Login = {
username: "pantao",
};
你们也看到了,Login 类型还是只有 username 是必填的,password 是可选的,这与我们的要求不符,怎么办?加上 Required 啊。
type User = {
username: string;
gender?: "male" | "female";
age?: number;
bio?: string;
password?: string;
};
type Login = Required<Pick<User, "username" | "password">>;
const login: Login = {
username: "pantao",
};
此时就会要求你必须输入 password 了。
如何自己实现 Pick?
/**
* From T, pick a set of properties whose keys are in the union K
*/
type Pick<T, K extends keyof T> = {
[P in K]: T[P];
};
假设我们现在在开发一个商城系统,每一个 Store 都有一个店长,一个仓库管理员,一个收银员,他们都关联到了 User 上面,此时你可以这样定义 Store 这个类型:
type User = {
username: string;
gender?: "male" | "female";
age?: number;
bio?: string;
password?: string;
};
type Store = {
name: string;
location: string;
leader: User;
warehouseKeeper: User;
cashier: User;
};
当然,你还可以这样定义:
type Store = Record<'leader' | 'warehouseKeeper' | 'cashier', User> & {
name: string;
location: string;
}
两种方式,哪种更好,当然各有优劣,但是假设我们遇到下面这样的一个情况:
type User = {
username: string;
gender?: "male" | "female";
age?: number;
bio?: string;
password?: string;
};
const users: User = [
{
username: "awesomeuser",
},
{
username: "baduser",
},
{
username: "gooduser",
},
];
为了访问方便,我想将 users 变量转换成一个属性名称为 users 数组索引,值为 users 数组元素(即 User 类型)的对象,怎么定义这样的一个类型?
你可以这样:
type UserSet = { [key: number]: User };
const userSet: UserSet = users.reduce(
(set, user, index) => ({ ...set, [index]: user }),
{} as UserSet
);
你也可以这样:
type UserSet = Record<number, User>;
如何自己实现 Record?
/**
* Construct a type with a set of properties K of type T
*/
type Record<K extends keyof any, T> = {
[P in K]: T;
};
近些日子,我使用了新语言编程,从JavaScript,切确地说是Elm,转成TypeScript。在本文中,我将继续深挖一些我非常喜欢的TypeScript特性。
TypeScript 和 JavaScript 是目前项目开发中较为流行的两种脚本语言,我们已经熟知 TypeScript 是 JavaScript 的一个超集,但是 TypeScript 与 JavaScript 之间又有什么样的区别呢?
Nerv_是一款由京东凹凸实验室打造的类 React 前端框架,基于虚拟 DOM 技术的 JavaScript(TypeScript) 库。它基于React标准,提供了与 React 16 一致的使用方式与 API。
交叉类型:将多个类型合并为一个类型、联合类型:表示取值可以为多种类型中的一种、混合类型:一个例子就是,一个对象可以同时做为函数和对象使用,并带有额外的属性、类型断言:可以用来手动指定一个值的类型
在做比较大的,多人合作的项目的时候,TypeScript会更加地适合,这得益于它的可读性,面向对象性以及易于重构的特点。但如果只是自己做小程序,不需要太多人参与的时候,JavaScript则会更加简单。
有两种方式安装TypeScript,如何创建第一个TypeScript文件,在TypeScript中,可以使用interface来描述一个对象有firstName和lastName两个属性,TypeScript支持JavaScript的新功能,其中很重要的一个功能就是基于类的面向对象编程
使用TypeScript已经有了一段时间,这的确是一个好东西,虽说在使用的过程中也发现了一些bug,不过都是些小问题,所以整体体验还是很不错的。有关TypeScript声明类型声明相关的目前就总结了这些比较常用的
谷歌在很早之前就张开双臂拥抱 Web 应用程序,Gmail 已经发布 14 年了。当时,JavaScript 的世界是疯狂的。Gmail 工程师不得不为 IE 糟糕的垃圾回收算法捏一把汗,他们需要手动将字符串文字从 for 循环中提取出来,以避免 GC 停顿
TypeScript是一种由微软开发的自由和开源的编程语言。它是JavaScript的一个超集,TypeScript是JavaScript类型的超集,它可以编译成纯JavaScript。TypeScript可以在任何浏览器、任何计算机和任何操作系统上运行,并且是开源的。
差不多两年前,我在一个创业团队中开始了一个全新的项目。用到的全都是类似Microservices,docker,react,redux这些时髦的东西。我在前端技术方面积累了一些类似的经验
内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!