Top Advanced typescript concepts that Every Developer Should Know

typescript 是一种现代编程语言,由于其附加的类型安全性,通常比 javascript 更受青睐。在本文中,我将分享 10 个最重要的 typescript 概念,这将有助于提高您的 typescript 编程技能。准备好了吗?开始吧。

Top Advanced typescript concepts that Every Developer Should Know

1.泛型:使用泛型我们可以创建可重用的类型,这将有助于处理今天和明天的数据。
泛型示例:
我们可能想要 typescript 中的一个函数将参数作为某种类型,并且我们可能想要返回相同的类型。

function func(args:t):t{    return args;}

2.具有类型约束的泛型:现在让我们通过定义它只接受字符串和整数来限制类型 t:

function func(value: t): t {    return value;}const stringvalue = func("hello"); // works, t is stringconst numbervalue = func(42);      // works, t is number// const booleanvalue = func(true); // error: type 'boolean' is not assignable to type 'string | number'

3.通用接口:
当您想要为使用各种类型的对象、类或函数定义契约(形状)时,接口泛型非常有用。它们允许您定义一个蓝图,该蓝图可以适应不同的数据类型,同时保持结构一致。

// generic interface with type parameters t and uinterface repository {    items: t[];           // array of items of type t    add(item: t): void;   // function to add an item of type t    getbyid(id: u): t | undefined; // function to get an item by id of type u}// implementing the repository interface for a user entityinterface user {    id: number;    name: string;}class userrepository implements repository {    items: user[] = [];    add(item: user): void {        this.items.push(item);    }     getbyid(idorname: number | string): user | undefined {        if (typeof idorname === 'string') {            // search by name if idorname is a string            console.log('searching by name:', idorname);            return this.items.find(user => user.name === idorname);        } else if (typeof idorname === 'number') {            // search by id if idorname is a number            console.log('searching by id:', idorname);            return this.items.find(user => user.id === idorname);        }        return undefined; // return undefined if no match found    }}// usageconst userrepo = new userrepository();userrepo.add({ id: 1, name: "alice" });userrepo.add({ id: 2, name: "bob" });const user1 = userrepo.getbyid(1);const user2 = userrepo.getbyid("bob");console.log(user1); // output: { id: 1, name: "alice" }console.log(user2); // output: { id: 2, name: "bob" }

4.泛型类::当您希望类中的所有属性都遵循泛型参数指定的类型时,请使用此选项。这允许灵活性,同时确保类的每个属性都与传递给类的类型匹配。

interface user {    id: number;    name: string;    age: number;}class userdetails {    id: t['id'];    name: t['name'];    age: t['age'];    constructor(user: t) {        this.id = user.id;        this.name = user.name;        this.age = user.age;    }    // method to get user details    getuserdetails(): string {        return `user: ${this.name}, id: ${this.id}, age: ${this.age}`;    }    // method to update user name    updatename(newname: string): void {        this.name = newname;    }    // method to update user age    updateage(newage: number): void {        this.age = newage;    }}// using the userdetails class with a user typeconst user: user = { id: 1, name: "alice", age: 30 };const userdetails = new userdetails(user);console.log(userdetails.getuserdetails());  // output: "user: alice, id: 1, age: 30"// updating user detailsuserdetails.updatename("bob");userdetails.updateage(35);console.log(userdetails.getuserdetails());  // output: "user: bob, id: 1, age: 35"console.log(new userdetails("30"));  // error: "this will throw error" 

5.将类型参数约束为传递的类型:有时,我们希望参数类型依赖于其他一些传递的参数。听起来很混乱,让我们看下面的示例。

function getproperty(obj: type, key: keyof type) {  return obj[key];}let x = { a: 1, b: 2, c: 3 };getproperty(x, "a");  // validgetproperty(x, "d");  // error: argument of type '"d"' is not assignable to parameter of type '"a" | "b" | "c"'.

6.条件类型:通常,我们希望我们的类型是一种类型或另一种类型。在这种情况下,我们使用条件类型。
一个简单的例子是:

function func(param:number|boolean){return param;}console.log(func(2)) //output: 2 will be printedconsole.log(func("true")) //error: boolean cannot be passed as argument

有点复杂的例子:

type hasproperty = k extends "age" ? "has age" : "has name";interface user {  name: string;  age: number;}let test1: hasproperty;  // "has age"let test2: hasproperty; // "has name"let test3: hasproperty; // error: type '"email"' is not assignable to parameter of type '"age" | "name"'.

6.交集类型:当我们想要将多种类型合并为一个类型时,这些类型非常有用,允许特定类型继承各种其他类型的属性和行为。
让我们看一个有趣的例子:

// defining the types for each area of well-beinginterface mentalwellness {  mindfulnesspractice: boolean;  stresslevel: number; // scale of 1 to 10}interface physicalwellness {  exercisefrequency: string; // e.g., "daily", "weekly"  sleepduration: number; // in hours}interface productivity {  taskscompleted: number;  focuslevel: number; // scale of 1 to 10}// combining all three areas into a single type using intersection typestype healthybody = mentalwellness & physicalwellness & productivity;// example of a person with a balanced healthy bodyconst person: healthybody = {  mindfulnesspractice: true,  stresslevel: 4,  exercisefrequency: "daily",  sleepduration: 7,  taskscompleted: 15,  focuslevel: 8};// displaying the informationconsole.log(person);

7.infer 关键字:当我们想要有条件地确定特定类型时,infer 关键字很有用,并且当条件满足时,它允许我们从该类型中提取子类型。
这是一般语法:

type conditionaltype = t extends sometype ? inferredtype : othertype;

示例:

type returntypeofpromise = t extends promise ? u : number;type result = returntypeofpromise<promise>;  // result is 'string'type errorresult = returntypeofpromise;      // errorresult is 'never'const result: result = "hello";console.log(typeof result); // output: 'string'

8.type variance:这个概念讨论了子类型和父类型如何相互关联。
它们有两种类型:
协方差:可以在需要父类型的地方使用子类型。
让我们看一个例子:

class vehicle { }class car extends vehicle { }function getcar(): car {  return new car();}function getvehicle(): vehicle {  return new vehicle();}// covariant assignmentlet car: car = getcar();let vehicle: vehicle = getvehicle(); // this works because car is a subtype of vehicle

在上面的示例中,car 继承了 vehicle 类的属性,因此将其分配给需要超类型的子类型是绝对有效的,因为子类型将具有超类型所具有的所有属性。
逆变:这与协变相反。我们在子类型应该出现的地方使用超类型。

class vehicle {  startengine() {    console.log("vehicle engine starts");  }}class car extends vehicle {  honk() {    console.log("car honks");  }}function processvehicle(vehicle: vehicle) {  vehicle.startengine(); // this works  // vehicle.honk(); // error: 'honk' does not exist on type 'vehicle'}function processcar(car: car) {  car.startengine(); // works because car extends vehicle  car.honk();        // works because 'car' has 'honk'}let car: car = new car();processvehicle(car); // this works because of contravariance (car can be used as vehicle)processcar(car);     // this works as well because car is of type car// contravariance failure if you expect specific subtype behavior in the method

使用逆变时,我们需要小心不要访问特定于子类型的属性或方法,因为这可能会导致错误。

9。反思: 这个概念涉及在运行时确定变量的类型。虽然 typescript 主要关注编译时的类型检查,但我们仍然可以利用 typescript 运算符在运行时检查类型。
typeof 运算符:我们可以利用 typeof 运算符在运行时查找变量的类型

const num = 23;console.log(typeof num); // "number"const flag = true;console.log(typeof flag); // "boolean"

instanceof 运算符:instanceof 运算符可用于检查对象是否是类或特定类型的实例。

class vehicle {  model: string;  constructor(model: string) {    this.model = model;  }}const benz = new vehicle("mercedes-benz");console.log(benz instanceof vehicle); // true

我们可以使用第三方库在运行时确定类型。

10.依赖注入:依赖注入是一种模式,允许您将代码引入组件中,而无需在组件中实际创建或管理它。虽然它看起来像是使用库,但它有所不同,因为您不需要通过 cdn 或 api 安装或导入它。

乍一看,它似乎也类似于使用函数来实现可重用性,因为两者都允许代码重用。然而,如果我们直接在组件中使用函数,可能会导致它们之间的紧密耦合。这意味着函数或其逻辑的任何变化都可能影响它使用的每个地方。

依赖注入通过将依赖项的创建与使用它们的组件解耦来解决这个问题,使代码更易于维护和测试。

没有依赖注入的示例

// health-related service classes without interfacesclass mentalwellness {  getmentalwellnessadvice(): string {    return "take time to meditate and relax your mind.";  }}class physicalwellness {  getphysicalwellnessadvice(): string {    return "make sure to exercise daily for at least 30 minutes.";  }}// healthadvice class directly creating instances of the servicesclass healthadvice {  private mentalwellnessservice: mentalwellness;  private physicalwellnessservice: physicalwellness;  // directly creating instances inside the class constructor  constructor() {    this.mentalwellnessservice = new mentalwellness();    this.physicalwellnessservice = new physicalwellness();  }  // method to get both mental and physical wellness advice  gethealthadvice(): string {    return `${this.mentalwellnessservice.getmentalwellnessadvice()} also, ${this.physicalwellnessservice.getphysicalwellnessadvice()}`;  }}// creating an instance of healthadvice, which itself creates instances of the servicesconst healthadvice = new healthadvice();console.log(healthadvice.gethealthadvice());// output: "take time to meditate and relax your mind. also, make sure to exercise daily for at least 30 minutes."

依赖注入示例

// Health-related service interfaces with "I" prefixinterface IMentalWellnessService {  getMentalWellnessAdvice(): string;}interface IPhysicalWellnessService {  getPhysicalWellnessAdvice(): string;}// Implementations of the servicesclass MentalWellness implements IMentalWellnessService {  getMentalWellnessAdvice(): string {    return "Take time to meditate and relax your mind.";  }}class PhysicalWellness implements IPhysicalWellnessService {  getPhysicalWellnessAdvice(): string {    return "Make sure to exercise daily for at least 30 minutes.";  }}// HealthAdvice class that depends on services via interfacesclass HealthAdvice {  private mentalWellnessService: IMentalWellnessService;  private physicalWellnessService: IPhysicalWellnessService;  // Dependency injection via constructor  constructor(    mentalWellnessService: IMentalWellnessService,    physicalWellnessService: IPhysicalWellnessService  ) {    this.mentalWellnessService = mentalWellnessService;    this.physicalWellnessService = physicalWellnessService;  }  // Method to get both mental and physical wellness advice  getHealthAdvice(): string {    return `${this.mentalWellnessService.getMentalWellnessAdvice()} Also, ${this.physicalWellnessService.getPhysicalWellnessAdvice()}`;  }}// Dependency injectionconst mentalWellness: IMentalWellnessService = new MentalWellness();const physicalWellness: IPhysicalWellnessService = new PhysicalWellness();// Injecting services into the HealthAdvice classconst healthAdvice = new HealthAdvice(mentalWellness, physicalWellness);console.log(healthAdvice.getHealthAdvice());// Output: "Take time to meditate and relax your mind. Also, Make sure to exercise daily for at least 30 minutes."

在紧密耦合的场景中,如果您今天 mentalwellness 类中有一个stresslevel 属性,并决定明天将其更改为其他属性,则需要更新所有使用它的地方。这可能会导致许多重构和维护挑战。

但是,通过依赖注入和接口的使用,你可以避免这个问题。通过构造函数传递依赖项(例如 mentalwellness 服务),具体的实现细节(例如stresslevel 属性)被抽象到接口后面。这意味着只要接口保持不变,对属性或类的更改不需要修改依赖类。这种方法可确保代码松散耦合、更易于维护且更易于测试,因为您可以在运行时注入所需的内容,而无需紧密耦合组件。

以上就是Top Advanced typescript concepts that Every Developer Should Know的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1497682.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 19:44:36
下一篇 2025年12月19日 19:44:46

相关推荐

  • Chrome扩展开发:图片资源加载与显示疑难解析

    本文详细探讨Chrome扩展程序中图片资源加载不显示的问题,重点解析`manifest.json`中`web_accessible_resources`配置的重要性,以及如何在JavaScript中利用`chrome.runtime.getURL()`动态获取扩展程序内部资源的正确URL。通过具体代…

    2025年12月20日
    000
  • 实现页面加载时播放启动画面,关闭标签页后重置

    本文介绍如何使用 JavaScript 实现一个在页面首次加载时播放启动画面,并在用户关闭标签页后重置的功能。我们将探讨如何利用 `sessionStorage` API 来存储会话状态,从而控制启动画面的显示逻辑,确保每次用户打开新标签页时都能看到启动画面。 使用 sessionStorage 控…

    2025年12月20日
    000
  • 处理动态表单数据:PHP 中访问动态添加的表单字段

    本文旨在解决在 PHP 中处理通过 JavaScript 动态生成的表单数据的问题。重点讲解如何正确命名表单字段,以便在 PHP 后端能够有效地访问和处理这些数据,并提供详细的代码示例和步骤说明,帮助开发者理解并实现动态表单数据的处理。 动态表单在 Web 开发中非常常见,尤其是在需要用户输入数量不…

    2025年12月20日
    000
  • Svelte组件间响应式变量管理的最佳实践

    本文旨在深入探讨Svelte中父子组件间响应式变量的正确管理与通信机制。我们将解析直接DOM操作和变量作用域不匹配导致的问题,并介绍如何利用Svelte的属性(props)、双向绑定(bind:)、事件分发(createEventDispatcher)以及指令(class:)等核心特性,实现高效、可…

    2025年12月20日
    000
  • JavaScript Tree Shaking死代码消除原理

    Tree Shaking 是一种基于 ES6 模块静态分析的构建优化技术,通过标记并剔除未使用的代码来减小打包体积;它依赖于静态的 import/export 语法,不支持 CommonJS 动态模块,在生产模式下结合 Terser 等压缩工具生效,且需在 package.json 中配置 side…

    2025年12月20日
    000
  • JavaScript函数柯里化与组合技术

    函数柯里化将多参数函数转换为单参数函数序列,如add(1)(2)(3);组合通过compose或pipe串联函数,实现声明式数据流处理,两者结合提升代码复用性与可读性。 函数柯里化和组合是函数式编程中的两个核心概念,在JavaScript中被广泛使用,尤其在处理高阶函数、构建可复用逻辑和提升代码表达…

    2025年12月20日
    000
  • 使用共享状态和Proxy模式管理多事件监听器间的逻辑依赖

    当多个事件监听器之间存在隐式逻辑依赖时,代码的可读性和维护性会显著下降。本文介绍一种通过共享状态对象来明确管理这些依赖的教程,特别是在处理如元素拖拽等复杂交互时。我们将演示如何利用javascript的proxy对象,以一种解耦且可控的方式,响应状态变化并执行相应的操作,从而构建结构清晰、易于理解的…

    2025年12月20日
    000
  • Svelte组件间通信与状态管理:解决父子组件响应式变量更新难题

    本文深入探讨svelte中父子组件间响应式变量更新的常见误区,如手动dom操作和不当的函数传递。通过详细讲解`bind:`指令实现双向绑定、`class:`指令管理css类以及`createeventdispatcher`实现事件通信,提供svelte最佳实践,帮助开发者构建高效、可维护的组件,避免…

    2025年12月20日
    000
  • Mongoose:无需Schema也能查询数据

    本文探讨了在mongoose中查询mongodb数据时是否必须先定义模型和schema的问题。我们将介绍如何利用mongoose的`connection.prototype.collection()`方法,直接获取mongodb驱动的原始集合实例,从而在不创建mongoose模型和schema的情况…

    2025年12月20日
    000
  • Svelte组件间状态同步与响应式更新指南

    在svelte应用开发中,一个常见的挑战是如何确保组件内部的响应式状态能够根据父组件的交互或数据变化而正确更新。当父组件通过直接操作dom来改变ui状态时,子组件的内部响应式变量往往不会随之更新,导致视图与数据不同步。理解svelte的响应式机制和组件间通信的最佳实践,是解决这类问题的关键。 Sve…

    2025年12月20日
    000
  • JavaScript迭代器与生成器原理

    迭代器是实现next方法并返回value和done的对象,用于遍历数据序列;2. 实现Symbol.iterator方法可使对象可迭代,支持for…of遍历;3. 生成器函数通过yield简化迭代器创建,自动提供next方法。 JavaScript中的迭代器和生成器是处理数据序列的重要工…

    2025年12月20日
    000
  • Solid.js 文件上传指南:解决后端接收空文件问题

    本文旨在解决 Solid.js 中使用 `createSignal` 或 `createStore` 进行多文件上传时,后端接收到空文件的问题。我们将详细介绍如何正确地使用 `createStore` 管理文件状态,并提供一个完整的前端示例,确保文件能够成功上传至后端。 在 Solid.js 中实现…

    2025年12月20日
    000
  • Chrome扩展开发:解决内容脚本中图片资源不显示问题

    本文详细探讨chrome扩展开发中,内容脚本(content script)无法正确显示图片资源的问题及解决方案。核心在于理解`chrome.runtime.geturl()`的作用,并正确配置`manifest.json`中的`web_accessible_resources`,确保图片路径在ja…

    2025年12月20日
    000
  • Node.js/MySQL动态批量更新多行数据的策略与实践

    在node.js应用中,使用`mysqljs/mysql`模块进行动态批量更新多行数据时,直接将多组数据作为单个`update`语句的参数会导致语法错误,因为其处理方式不同于批量`insert`。本文将深入探讨这一常见误区,并提供三种有效的解决方案:利用`insert … on dupl…

    2025年12月20日
    000
  • TypeScript在大型JavaScript项目中的应用

    TypeScript通过静态类型系统提升大型JavaScript项目的可维护性、协作效率与稳定性,支持渐进式迁移和生态兼容,结合泛型、接口等设计优化架构,显著降低维护成本。 TypeScript 在大型 JavaScript 项目中的应用越来越广泛,主要原因在于它为 JavaScript 增加了静态…

    2025年12月20日
    000
  • 处理动态表单:在PHP中访问和存储动态生成的表单数据

    本文档旨在解决在PHP中处理动态生成的HTML表单数据的问题。我们将探讨如何通过JavaScript动态创建表单元素,并确保这些元素在提交后能够被PHP正确接收和处理。重点在于理解如何使用数组形式的name属性来组织表单数据,以便在PHP中轻松访问和存储这些数据,同时避免因重复ID导致的潜在问题。 …

    2025年12月20日
    000
  • Mongoose:无需定义模型,直接查询MongoDB集合数据

    在mongoose中,通常需要定义模型和schema来操作数据。然而,本文将探讨如何在不定义mongoose模型和schema的情况下,直接访问并查询mongodb集合。通过使用`connection.prototype.collection()`方法,开发者可以直接获取原生mongodb驱动的集合…

    2025年12月20日
    000
  • 处理动态表单数据:PHP 接收并处理 JavaScript 动态生成的表单项

    本文旨在解决如何使用 PHP 正确接收并处理 JavaScript 动态生成的表单数据的问题。通过修改 HTML 结构,利用数组命名规则,并结合 JavaScript 动态生成表单元素,最终实现 PHP 后端对动态表单数据的有效处理和存储。重点在于理解 HTML 表单的 `name` 属性,以及如何…

    2025年12月20日
    000
  • CSS z-index 属性在复杂布局中解决图片遮挡问题的实践

    本文详细探讨了在复杂的css布局(如家谱树)中,当悬停弹窗图片被相邻元素遮挡时,如何利用`z-index`属性有效解决这一问题。通过分析堆叠上下文和`position`属性,教程提供了具体的css代码修改示例,并强调了使用`z-index`的关键注意事项,确保图片能正确显示在其他元素之上。 解决复杂…

    2025年12月20日
    000
  • Svelte组件通信与状态管理:解决父子组件响应式更新问题的最佳实践

    本文深入探讨svelte中父子组件通信和状态管理的常见误区,特别是避免手动dom操作和理解组件作用域的重要性。通过详细介绍svelte的props、`bind:`指令、事件派发器及`class:`指令,指导开发者构建高效、响应式的svelte应用,确保组件间数据流的正确更新。 在Svelte应用开发…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信