Typescript高级: 深入理解 keyof 和 extends keyof

2024-05-12 17:12

本文主要是介绍Typescript高级: 深入理解 keyof 和 extends keyof,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

  • 在TypeScript的世界中,extends和keyof是两个强大的工具
  • 它们分别用于约束类型和获取对象类型的键
  • 当这两者结合使用时,我们可以创建出更为复杂和精细的类型操作
  • 从而增强TypeScript的类型安全性

keyof

  • keyof 用于获取一个对象类型的所有键,并生成一个新的字符串字面量类型

    type Person = {name: string;age: number;address: {city: string;state: string;};};type PersonKeys = keyof Person; // "name" | "age" | "address"const a: PersonKeys = "name" // 正确
    const b: PersonKeys = "age" // 正确
    const c: PersonKeys = "address" // 正确
    // const d: PersonKeys = "xxxx" // 编译就错误console.log(a) // name
    console.log(b) // age
    console.log(c) // address
    
  • 在上面的代码中,PersonKeys是一个由Person对象的所有键组成的联合类型

    const obj = { address: "aaaaaaa", phone: 153111111, descri: "bbbbbb" }
    console.log(typeof obj) // object// 1 ) 单个的 keyof 测试
    type myobjtype = typeof obj // 这里 typeof obj 是 object, 所以 myobjtype 就是 object
    type keyofobj = keyof myobjtype // "address" | "phone" | "descri"
    const a:keyofobj = "address"
    const b:keyofobj = "phone"
    const c:keyofobj = "descri"
    console.log(a) // address
    console.log(b) // phone
    console.log(c) // descriconsole.log("----------------------")// 2 ) 组合 keyof 和 typeof 效果同上
    type keyofobjtype = keyof typeof obj;
    const d: keyofobjtype = "address"
    const e: keyofobjtype = "phone"
    const f: keyofobjtype = "descri"
    console.log(d) // address
    console.log(e) // phone
    console.log(f) // descri// 3 ) 定义 type 与 获取type
    type testType = { username: string, age: number }
    type ageType = testType["age"]
    type nameType = testType["username"]
    const g : ageType = 1 // 数值
    const h: nameType = "abc"
    console.log(g) // 1
    console.log(h) // abc// 4 )定义 inteface 和 结合 keyof使用interface
    interface objType { username: string, age: number }
    const i: objType = { username: "博鳌", age: 1111 }
    type xxType = keyof objType
    const j: xxType = "username"
    const k: xxType = "age"
    console.log(j) // username
    console.log(k) // age
    type yType = objType["age"]
    const l:yType = 11
    console.log(l) // 11
    
  • keyof 对于类的使用

    class Order {static count: numberconstructor(public orderId: number, public date: Date,public custname: string,public phone: string) {}doEat() {}
    }type keyofOrders = keyof Order // "orderId" | "date" | "custname" | "phone" | "doEat"
    const a: keyofOrders = "orderId"
    const b: keyofOrders = "date"
    const c: keyofOrders = "custname"
    const d: keyofOrders = "phone"
    const e: keyofOrders = "doEat"
    console.log(a) // orderId
    console.log(b) // date
    console.log(c) // custname
    console.log(d) // phone
    console.log(e) // doEat
    

extends 约束

  • extends关键字在TypeScript中通常用于类型约束
  • 确保一个类型符合特定的条件或结构
  • 例如,在泛型中,我们可以使用extends来限制类型参数的范围
    function printLength<T extends { length: number }>(obj: T): void {console.log(obj.length);
    }printLength({ length: 10, value: "Hello" }); // 10 这里正常编译
    // printLength({ value: "Hello" }); // Error: Object literal may only specify known properties, and 'value' does not exist in type '{ length: number; }'.
    
  • 上述定义了一个 printLength ,参数是 obj, 参数类型是 T, 返回值是 void
  • 对于参数类型T来说,需要符合一种类型,简单来说就是继承 length: number 这个对象
  • 而下面传入的参数中不符合参数要继承 length: number 的这一约束
  • 我们确保传递给该函数的任何对象都必须具有一个名为length的数字类型属性

extends keyof 结合使用

  • 当extends和keyof结合使用时,我们可以创建一个类型
  • 该类型确保某个键存在于给定的对象类型中
  • 这通常在类型安全和映射类型中非常有用
  • 以下是一个使用extends keyof的例子,它定义了一个函数
  • 该函数接受一个对象和一个该对象的键,并返回该键对应的值类型:
    type ValueOfType<T, K extends keyof T> = T[K];function getValue<T, K extends keyof T>(obj: T, key: K): ValueOfType<T, K> {return obj[key];
    }const person = {name: "Alice",age: 30,
    };const nameValue: string = getValue(person, "name"); // Alice
    const ageValue: number = getValue(person, "age"); // 30
    console.log(nameValue)
    console.log(ageValue)
    // const invalidValue: string = getValue(person, "invalidKey"); // Error: Argument of type '"invalidKey"' is not assignable to parameter of type '"name" | "age"'.
    
  • 在这个例子中,ValueOfType是一个泛型类型,它接受两个类型参数:T和K
    • K通过extends keyof T约束,确保它必须是T对象的键之一
    • ValueOfType<T, K>返回T[K],即对象T中键为K的属性的类型
  • getValue函数使用ValueOfType来确保返回值的类型与提供的键匹配
  • 这里,getValue<T, K extends keyof T>
    • 后面一半的泛型 <T, K extends keyof T>
    • 用于描述这个函数,可以理解为函数的描述
  • 后面就是具体的应用示例了
  • 最后如果尝试使用不存在的键调用getValue,TypeScript编译器会抛出错误

总结

  • 通过结合使用extends和keyof,可以在TS中创建更为精细和安全的类型操作
  • 这种组合使我们能够确保类型参数的键存在于对象中,并据此推断出相应值的类型
  • 这不仅增强了代码的类型安全性,还提高了代码的可读性和可维护性

这篇关于Typescript高级: 深入理解 keyof 和 extends keyof的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/983235

相关文章

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

从基础到高级详解Python数值格式化输出的完全指南

《从基础到高级详解Python数值格式化输出的完全指南》在数据分析、金融计算和科学报告领域,数值格式化是提升可读性和专业性的关键技术,本文将深入解析Python中数值格式化输出的相关方法,感兴趣的小伙... 目录引言:数值格式化的核心价值一、基础格式化方法1.1 三种核心格式化方式对比1.2 基础格式化示例

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

深度解析Python yfinance的核心功能和高级用法

《深度解析Pythonyfinance的核心功能和高级用法》yfinance是一个功能强大且易于使用的Python库,用于从YahooFinance获取金融数据,本教程将深入探讨yfinance的核... 目录yfinance 深度解析教程 (python)1. 简介与安装1.1 什么是 yfinance?

MySQL数据类型与表操作全指南( 从基础到高级实践)

《MySQL数据类型与表操作全指南(从基础到高级实践)》本文详解MySQL数据类型分类(数值、日期/时间、字符串)及表操作(创建、修改、维护),涵盖优化技巧如数据类型选择、备份、分区,强调规范设计与... 目录mysql数据类型详解数值类型日期时间类型字符串类型表操作全解析创建表修改表结构添加列修改列删除列

深入理解go中interface机制

《深入理解go中interface机制》本文主要介绍了深入理解go中interface机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前言interface使用类型判断总结前言go的interface是一组method的集合,不

Python 函数详解:从基础语法到高级使用技巧

《Python函数详解:从基础语法到高级使用技巧》本文基于实例代码,全面讲解Python函数的定义、参数传递、变量作用域及类型标注等知识点,帮助初学者快速掌握函数的使用技巧,感兴趣的朋友跟随小编一起... 目录一、函数的基本概念与作用二、函数的定义与调用1. 无参函数2. 带参函数3. 带返回值的函数4.

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer

Java Stream 的 Collectors.toMap高级应用与最佳实践

《JavaStream的Collectors.toMap高级应用与最佳实践》文章讲解JavaStreamAPI中Collectors.toMap的使用,涵盖基础语法、键冲突处理、自定义Map... 目录一、基础用法回顾二、处理键冲突三、自定义 Map 实现类型四、处理 null 值五、复杂值类型转换六、处理