深入探索 TypeScript:从基础到高级特性

深入探索 TypeScript:从基础到高级特性

一、引言

在现代软件开发领域,TypeScript 已经成为了一种极具影响力的编程语言。它基于 JavaScript,并为其添加了强大的静态类型系统,使得代码在开发阶段就能进行更严格的类型检查,从而减少运行时错误,提高代码的质量和可维护性。无论是大型企业级应用还是小型的前端项目,TypeScript 都展现出了独特的优势。在这篇博客中,我们将深入探讨 TypeScript 的各个方面,包括它的安装、类型系统、接口、交叉类型、断言、泛型、装饰器和重载等特性,让你全面了解这一强大的编程工具。

二、TypeScript 的安装与初体验

(一)安装 TypeScript

要开始使用 TypeScript,首先需要在本地环境中安装它。使用 Node Package Manager(npm)可以轻松完成安装。在命令行中执行以下命令:

npm install -g typescript

这条命令会在全局环境中安装 TypeScript。安装完成后,可以通过以下命令检查 TypeScript 的版本:

tsc -v

这一步确保 TypeScript 已经正确安装在你的系统中,为后续的开发工作做好准备。

三、TypeScript 的类型系统

(一)基本类型

TypeScript 提供了丰富的基本类型,包括booleanstringnumberarraynullundefinedobject。这些基本类型构成了 TypeScript 类型系统的基础。

let isDone: boolean = false;
let str: string = 'hello';
let num: number = 123;
let u: undefined = undefined;
let n: null = null;
let arr: number[] = [1, 2, 3];
let strArr: string[] = ['a', 'b'];
let list: any[] = [1, true, 'hello'];
let x: [string, number] = ['hello', 10];
let StrArr: Array<string> = ['a', 'b'];

在上述代码中,我们可以看到不同基本类型的变量声明方式。boolean类型用于表示真假值,string类型用于存储文本数据,number类型用于表示数字。对于数组类型,可以使用类型[]或者Array<类型>的方式来声明。而nullundefined在 TypeScript 中有明确的类型定义,它们与其他类型的使用方式有所不同。

(二)元组(Tuple)

元组是一种特殊的数据类型,它允许我们表示一个已知元素数量和类型的数组,每个元素的类型可以不同。

let tupleType: [string, number] = ['hello', 10];

在这个例子中,tupleType是一个包含一个字符串和一个数字的元组。元组在处理一些具有特定结构的数据时非常有用,例如函数返回多个不同类型的值时,可以使用元组来接收。

(三)枚举(Enum)

枚举是一种将一组相关的命名常量组织在一起的方式,它为代码增加了可读性和可维护性。

  1. 数字枚举
    数字枚举默认从 0 开始依次递增。
enum Color {Red, Green, Blue};
let color: Color = Color.Green;

在这个例子中,Color枚举定义了三个颜色常量,Red的值为 0,Green的值为 1,Blue的值为 2。

  1. 字符串枚举
    除了数字枚举,还可以定义字符串枚举。
enum Color {Red = 'red', Green = 'green', Blue = 'blue'};
let color: Color = Color.Green;

字符串枚举使得每个枚举成员的值都明确指定为一个字符串,这种方式在代码即文档方面有很大优势,代码的可读性更强。

(四)特殊类型:any、unknown、void 和 never

  1. any 类型
    any类型是一种可以代表任意类型的值。它允许我们绕过类型检查,但过度使用any可能会导致类型系统的优势丧失。
let anyValue: any; // 任意类型 绕过 类型检查
anyValue = 'hello';
anyValue = 123;
anyValue = true;
anyValue = {};
  1. unknown 类型
    unknown类型表示一个未知类型的值。与any不同,unknown类型更加严格,在使用unknown类型的值时,需要先进行类型检查或断言。
let unknownValue: unknown;
unknownValue = 'hello';
// 以下代码会报错,因为 unknown 类型的值不能直接调用方法
// unknownValue.trim();
unknownValue = 123;
unknownValue = true;
unknownValue = {};
  1. any 和 unknown 的区别
    虽然anyunknown都可以绕过类型检查,但它们的严格程度不同。any可以赋值给任意类型,而unknown只能赋值给自身和any类型。

  2. void 类型
    void类型通常用于声明函数的返回值类型为空。

function warnUser(): void {console.log('This is a warning message');
}
  1. never 类型
    never类型用于声明函数永远不会返回值,例如抛出异常的函数。
function error(message: string): never {throw new Error(message);
}

voidnever的区别在于,void表示函数没有返回值(可以理解为返回undefined),而never表示函数根本不会正常返回。

(五)Object 类型

Object类型在 TypeScript 中有特定的含义,它代表标准对象,即非原始类型。

// 标准对象 => 非原始类型
interface MyObject {create(i: object | null): any
}
// Object.prototype

这里的object类型用于描述一个非原始类型的值,它可以是任何对象,包括通过new关键字创建的实例对象、对象字面量等。

四、接口(Interface)

(一)接口的基本概念

接口是 TypeScript 中一种重要的类型定义方式,它用于对行为的抽象,具体的行为则交给类来实现。接口定义了一组属性和方法的签名,类必须实现这些接口中定义的内容。

interface Class {name: string;time: number;
}
let classObj: Class = {name: 'TS',time: 2024
};

在这个例子中,Class接口定义了nametime两个属性,classObj对象实现了这个接口,它必须包含nametime属性,并且类型要与接口中定义的一致。

(二)接口的特性

  1. 只读属性(readonly)和可选属性(?)
    接口可以定义只读属性和可选属性。只读属性在初始化后不能被修改,可选属性则表示该属性在实现接口时可以存在也可以不存在。
interface Point {readonly x: number; // readonly 只读属性readonly y: number;z?: number; // 可选属性
}
let p1: Point = { x: 10, y: 20 };
p1.x = 5; // error!
  1. 面试:readonly vs const
    constreadonly都可以用于声明常量,但它们有一些区别。const声明常量后,必须赋值,且不能修改。
const num = 10;
num = 20; // error!
const obj = { a: 1, b: 2 };
obj.a = 3; // 这里对于对象内部属性的修改是允许的,因为 const 只是保证对象的引用不变

对于数组,我们可以通过ReadonlyArray类型来创建只读数组。

let arr: number[] = [1, 2];
let roArr: ReadonlyArray<number> = arr;
roArr[0] = 1; // error!
roArr.push(3); // error!
roArr.length = 1; // error!
arr = roArr; // error!
arr = [1, 2, 3]; // ok
roArr = arr; // ok
  1. 可添加性
    接口还可以定义任意属性,使用[propName: string]: any;的方式。
interface Point {x: number;y?: number;[propName: string]: any; // 任意属性
}
let p1: Point = { x: 10, z: 20, name: '坐标点位', color: 'red', err: '错误信息' };

这种方式使得接口在处理具有动态属性的对象时更加灵活,但需要注意避免滥用,以免影响类型的确定性。

五、交叉类型(&)

(一)交叉类型的概念

交叉类型允许我们将多个类型合并成一个类型,新的类型将包含所有参与交叉的类型的属性。

interface A {a: number;
}interface B {b: string;
}type AB = A & B; // 交叉类型
let ab: AB = { a: 1, b: 'hello' };

在这个例子中,AB类型是AB类型的交叉,ab对象必须同时满足AB接口的定义。

(二)更复杂的交叉类型示例

交叉类型可以用于更复杂的场景,例如多个接口之间的交叉。

interface A { x: D }
interface B { x: E }
interface C { x: F }interface D { d: number }
interface E { d: string }
interface F { d: boolean }type ABC = A & B & C;
let abc: ABC = { x: { d: true, e: 'hello', f: 10 } };

这里ABC类型是ABC三个接口的交叉,abc对象需要满足所有相关接口对于x属性的定义,虽然这个例子比较复杂,但展示了交叉类型在处理复杂类型关系时的强大能力。

(三)type(类型别名)vs interface(类型描述)

  1. 相同点
    两者都可以声明类型,并且都具有一定的可拓展性。

  2. 不同点

    • 声明次数interface可以声明多次,同名的interface会自动合并。例如:
interface Person {name: string;
}
interface Person {age: number;sex: string;
}

type不能声明两次,如果重复声明会报错。

- **修改限制**:`type`声明后,不能再次修改,它更像是一个一次性定义的类型别名。而`interface`在一定程度上可以通过多次声明来扩展。- **使用场景**:`type`更偏向于多重类型的动态计算,例如联合类型、交叉类型等复杂的类型操作。`interface`更偏向于描述对象的静态计算,因为它通常用于向外暴露类型定义,并且可以方便地进行扩展。

(四)面试 2:联合冲突

当使用交叉类型时,可能会遇到联合冲突的情况。

interface A {a: number;b: number;
}interface B {b: string;c: boolean;
}type AB = A & B;
let ab: AB = { a: 1, c: true };
// b: never 
// 因为没有一种类型可以同时满足 A 和 B,所以 b 为 never

在这种情况下,由于ABb属性的类型定义不一致,且没有共同的类型满足这两个接口,所以b的类型被推断为never

六、类型断言

(一)类型断言的概念

类型断言是一种告诉 TypeScript 编译器我们比它更了解某个值的类型的方式,它允许我们将一个值视为特定的类型,从而实现类型变量的多样化。

  1. as 语法
    使用as关键字进行类型断言。
let someValue: any = 'this is a string';
let strLength: number = (someValue as string).length;
  1. 尖括号语法(在某些情况下可能受限)
    在某些环境中,也可以使用尖括号的方式进行类型断言,但在 React 等一些 JavaScript 框架中,尖括号可能会与 JSX 语法冲突,所以as语法更为常用。
let someValue: any = 'this is a string';
let strLength: number = (<string>someValue).length;
  1. 非空断言
    在处理可能为undefinednull的值时,可以使用非空断言。
type ClassTime = () => string | number;
const start = (classTime: ClassTime | undefined) => {let number = classTime!(); // 非空断言
};

(二)类型守卫

类型守卫是一种在代码中进行类型检查的机制,它可以保证在语法规定的多种类型范围内做校验。

interface Teacher {name: string;age: number;courses: string[];
}
interface Person {name: string;age?: number;
}function getInfo(person: Person | Teacher): string {// 类型守卫 1// if ((person as Teacher).courses!== undefined) {//   return (person as Teacher).name + '教' + (person as Teacher).courses.join('');// } else {//   return (person as Person).name;// }// 类型守卫 2if ('courses' in person) {return person.name + '教' + (person as Teacher).courses.join('');} else {return (person as Person).name;}
}

在这个例子中,我们通过in关键字来检查person对象是否具有courses属性,从而判断personTeacher类型还是Person类型。这种类型守卫的方式使得我们可以在处理联合类型的值时,根据不同的类型执行相应的逻辑。

七、泛型

(一)泛型的概念

泛型是 TypeScript 中一个非常强大的特性,它主要解决类、接口、方法的复用性问题,以及对不特定数据类型的支持。泛型允许我们编写可以在多种类型上工作的代码,而不是针对特定类型编写重复的代码。

function getData<T, U>(value: T, score: U): T {return value;// 类型推断
}
let data = getData<string, number>('hello', 123);

getData函数中,TU是泛型类型参数。当我们调用getData函数时,可以指定TU的具体类型,这样函数就可以根据传入的类型进行相应的类型检查和操作。同时,TypeScript 也具有类型推断的能力,在一些情况下可以自动推断出泛型的类型。

(二)泛型函数的更多示例

泛型函数可以有更复杂的形式和应用场景。

function getData<T, U>(value: T, score: U): String {return value;
}function getData<T, U>(value: T, score: U): String {return (String(value)) as any as T;
}

这些示例展示了泛型函数在处理不同类型参数和返回值类型时的灵活性,但需要注意在使用类型转换时要确保类型的安全性,避免出现意外的类型错误。

八、装饰器(Decorator)

(一)装饰器的概念

装饰器是一种特殊的函数,它可以用来修改类、方法或属性。装饰器提供了一种简洁的方式来为代码添加额外的功能或行为。

function log(target: Function): void {// 对 target 进行加工console.log(target);target.prototype.sayHello = function () {console.log('hello');}
}function nameWrapper(target: any, key: string): void {Object.defineProperty(target, key, {// setter// getter})
}
@log
class Person {name: string;age: number;constructor(name: string, age: number) {@nameWrapperthis.name = name;this.age = age;console.log('Person');console.log(this);console.log(this.name, this.age);console.log(this instanceof Person);}
}

在这个例子中,log装饰器函数接受一个类的构造函数作为参数,并为该类添加了一个sayHello方法。nameWrapper装饰器函数则可以用于修改类的属性的描述符,例如添加settergetter方法。通过使用装饰器,我们可以在不修改类的原始代码的情况下,为类添加新的功能。

(二)装饰器的应用场景

装饰器在许多场景中都有广泛的应用,比如日志记录、性能监测、权限验证等。例如,在一个 Web 应用程序中,可以使用装饰器来记录某个方法的调用时间,或者检查用户是否有执行某个方法的权限。通过将这些功能封装在装饰器中,可以使代码更加清晰和可维护,将业务逻辑与这些额外的功能分离开来。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/471814.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

云计算研究实训室建设方案

一、引言 随着云计算技术的迅速发展和广泛应用&#xff0c;职业院校面临着培养云计算领域专业人才的迫切需求。本方案旨在构建一个先进的云计算研究实训室&#xff0c;为学生提供一个集理论学习、实践操作、技术研发与创新于一体的综合性学习平台&#xff0c;以促进云计算技术…

通过Python 调整Excel行高、列宽

在Excel中&#xff0c;默认的行高和列宽可能不足以完全显示某些单元格中的内容&#xff0c;特别是当内容较长时。通过调整行高和列宽&#xff0c;可以确保所有数据都能完整显示&#xff0c;避免内容被截断。合理的行高和列宽可以使表格看起来更加整洁和专业&#xff0c;尤其是在…

【代码审计】常见漏洞专项审计-业务逻辑漏洞审计

❤️博客主页&#xff1a; iknow181 &#x1f525;系列专栏&#xff1a; 网络安全、 Python、JavaSE、JavaWeb、CCNP &#x1f389;欢迎大家点赞&#x1f44d;收藏⭐评论✍ 0x01 漏洞介绍 1、 原理 业务逻辑漏洞是一类特殊的安全漏洞&#xff0c;业务逻辑漏洞属于设计漏洞而非实…

Wordpress常用配置,包括看板娘跨域等

一个Wordpress的博客已经搭建完成了&#xff0c;那么为了让它看起来更有人间烟火气一点&#xff0c;有一些常用的初始配置&#xff0c;这里整理一下。 修改页脚 页脚这里默认会显示Powered by Wordpress&#xff0c;还有一个原因是这里要加上备案信息。在主题里找到页脚&…

数据库范式、MySQL 架构、算法与树的深入解析

一、数据库范式 在数据库设计中&#xff0c;范式是一系列规则&#xff0c;用于确保数据的组织和存储具有良好的结构、完整性以及最小化的数据冗余。如果不遵循范式设计&#xff0c;数据可能会以平铺式罗列&#xff0c;仅使用冒号、分号等简单分隔。这种方式存在诸多弊端&#…

Taro React-Native IOS 打包发布

http网络请求不到 配置 fix react-native facebook::flipper::SocketCertificateProvider‘ (aka ‘int‘) is not a function or func_rn运行debug提示flipper-CSDN博客 Xcode 15&#xff08;iOS17&#xff09;编译适配报错_no template named function in namespace std-CS…

基于yolov8、yolov5的车型检测识别系统(含UI界面、训练好的模型、Python代码、数据集)

摘要&#xff1a;车型识别在交通管理、智能监控和车辆管理中起着至关重要的作用&#xff0c;不仅能帮助相关部门快速识别车辆类型&#xff0c;还为自动化交通监控提供了可靠的数据支撑。本文介绍了一款基于YOLOv8、YOLOv5等深度学习框架的车型识别模型&#xff0c;该模型使用了…

从python源码到可自动更新软件

相关阅读 标题链接如何打包python程序为exebczl【auto-py-to-exe 可视化打包python到exe】51CTO ZATA 1. python源码 打包时需要特别注意的源码编写规范 除了基本的 Python 编码规范之外,在准备程序进行打包时,还需要特别注意以下几点: 1.1 依赖管理 确保 requirements.t…

javaWeb小白项目--学生宿舍管理系统

目录 一、检查并关闭占用端口的进程 二、修改 Tomcat 的端口配置 三、重新启动 Tomcat 一、javaw.exe的作用 二、结束javaw.exe任务的影响 三、如何判断是否可以结束 结尾&#xff1a; 这个错误提示表明在本地启动 Tomcat v9.0 服务器时遇到了问题&#xff0c;原因是所需…

k8s笔记——核心概念

什么是K8s Kubernetes 也称为 K8s&#xff0c;是用于自动部署、扩缩和管理容器化应用程序的开源系统。 Kubernetes 最初是由 Google 工程师作为 Borg 项目开发和设计的&#xff0c;后于 2015 年捐赠给 云原生计算基金会&#xff08;CNCF&#xff09;。 什么是 Kubernetes 集群…

SkyWalking-安装

SkyWalking-简单介绍 是一个开源的分布式追踪系统&#xff0c;用于检测、诊断和优化分布式系统的功能。 支持 ElasticSearch、H2、MySQL、PostgreSql 等数据库 基于 ElasticSearch 的情况 ElasticSearch&#xff08;ES&#xff09; 安装 1、下载并解压 https://www.elastic…

怎么样绑定域名到AWS(亚马逊云)服务器

1&#xff0c;拿着你买的域名去亚马逊申请一个证书。申请证书分两种&#xff0c;一种是去亚马逊后台填域名手动申请 &#xff0c;另一种是通过API来申请&#xff0c;类似如下代码&#xff1a; 2、证验证书。有两种方式&#xff1a;一种是通过邮件&#xff0c;另一种去到域名提供…

C++常用的新特性-->day06

时间间隔duration duration表示一段时间间隔&#xff0c;用来记录时间长度&#xff0c;可以表示几秒、几分钟、几个小时的时间间隔。duration的原型如下 // 定义于头文件 <chrono> template<class Rep,class Period std::ratio<1> > class duration;Rep&…

云服务器端口开放

云服务器一般除了必要端口&#xff0c;其它端口默认都是关闭的&#xff0c;要想进行网络通信就必须开放指定的端口。下面以腾讯云服务器为例来进行详细说明。 腾讯云的轻量服务器是没有安全组的&#xff0c;这里有的只是防火墙&#xff0c;开放端口也是通过防火墙来设置的。首先…

[Linux网络编程]10-http协议,分别使用epoll和libevent两种方式实现B/S服务器

一.HTTP协议&#xff0c;Request请求和Response响应 二.常用状态码 接下来使用两种方式构建B/S模型,实现浏览器向客户端请求文件的功能。 浏览器向构建好的服务器发送请求&#xff0c;在请求行中请求具体文件&#xff0c;服务器连通客户端(浏览器)后&#xff0c;检查文件是否存…

C#从入门到放弃

C#和.NET的区别 C# C#是一个编程语言 .NET .NET是一个在window下创建程序的框架 .NET框架不仅局限于C#,它还可以支持很多语言 .NET包括了2个组件&#xff0c;一个叫CLR(通用语言运行时)&#xff0c;另一个是用来构建程序的类库 CLR 用C写一个程序&#xff0c;在一台8688的机器…

Suricata

02-Suricata 一 ICMP流量预警 一条ICMP报文有四个重要内容&#xff0c;可与相应的ICMP关键字相匹配。它们是&#xff1a;消息的类型、代码、ID和序列。 通过ICMP的type进行匹配 alert icmp any any <> any any (msg:"icmp流量预警";itype:8;threshold:type t…

在 WPF 中,如何实现数据的双向绑定?

在 WPF 中&#xff0c;数据绑定是一个非常重要的特性&#xff0c;它允许 UI 与数据源之间自动同步。双向绑定是一种常见的绑定方式&#xff0c;当数据源更新时&#xff0c;UI 会自动更新&#xff1b;同样&#xff0c;当 UI 中的元素&#xff08;如文本框&#xff09;发生改变时…

2、 家庭网络发展现状

上一篇我们讲了了解家庭网络历史(https://blog.csdn.net/xld_hung/article/details/143639618?spm1001.2014.3001.5502),感兴趣的同学可以看对应的文章&#xff0c;本章我们主要讲家庭网络发展现状。 关于家庭网络发展现状&#xff0c;我们会从国内大户型和小户型的网络说起&…

Linux——Linux环境基础开发工具使用

一、软件包管理器yum 在Linux上想要安装一个软件需要下载程序的源代码并且进行编译得到一个可执行的程序。这样会引发很多的问题&#xff0c;第一是这样做很麻烦&#xff1b;第二是我们不知道软件的开发者在开发这款软件的时候用的是一个什么样的环境&#xff0c;我们用不一样的…