TypeScript实用技巧大杂烩,助你成为真正的全栈工程师
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(255, 255, 255, 1); color: rgba(0, 0, 0, 1) }
.hljs-comment, .hljs-quote, .hljs-variable { color: rgba(0, 128, 0, 1) }
.hljs-built_in, .hljs-keyword, .hljs-name, .hljs-selector-tag, .hljs-tag { color: rgba(0, 0, 255, 1) }
.hljs-addition, .hljs-attribute, .hljs-literal, .hljs-section, .hljs-string, .hljs-template-tag, .hljs-template-variable, .hljs-title, .hljs-type { color: rgba(163, 21, 21, 1) }
.hljs-deletion, .hljs-meta, .hljs-selector-attr, .hljs-selector-pseudo { color: rgba(43, 145, 175, 1) }
.hljs-doctag { color: rgba(128, 128, 128, 1) }
.hljs-attr { color: rgba(255, 0, 0, 1) }
.hljs-bullet, .hljs-link, .hljs-symbol { color: rgba(0, 176, 232, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }
作为一个全栈工程师,精通一门语言是远远不够的。在 JavaScript 方式盛行的当下,TypeScript 更是成为了众多前端开发的选择。TypeScript 不仅有更好的开发体验,还能提高代码的健壮性,减少错误和编码时间。在本文中,我们将介绍 TypeScript 中一些实用技巧,助你成为真正的全栈工程师!
一、声明文件的使用技巧
声明文件的引入方式
声明文件可以让 TypeScript 对库、框架等第三方代码进行类型检查,提供更好的开发体验和代码可读性。当我们使用其他开源类库时,需要在项目中引入其对应的声明文件。 推荐使用 npm 安装库时自动下载声明文件的方式,例如我们在使用 lodash 库的时候,在命令行中输入:
npm install lodash --save-dev
npm install @types/lodash --save-dev
使用 namespace 保持代码干净整洁
当我们使用多个第三方库时,可能存在名称冲突的问题。这时可以使用 namespace 来避免冲突:
namespace MyUtils {
export function toArray<T>(args: IArguments): T[] {
return Array.prototype.slice.call(args);
}
}
namespace MyUtils2 {
export function toArray<T>(args: Array<T>): T[] {
return Array.from(args);
}
}
// 调用 toArray 方法
const arr1 = MyUtils.toArray(arguments);
const arr2 = MyUtils2.toArray([1, 2, 3]);
二、使用装饰器提升代码的可读性和可维护性
类的装饰器
装饰器是一种特殊的声明,它可以被附加到类声明、方法、属性或参数上,以修改类的行为。常见的类装饰器有:
// 简单的类装饰器
function myClassDecorator(target: any) {
// 修改类的行为、增加方法等
}
// 携带参数的类装饰器
function myClassDecoratorFactory(name: string) {
return function (target: any) {
// 修改类的行为、增加方法等
};
}
@myClassDecorator
class MyClass {}
@myClassDecoratorFactory('foo')
class MyClass2 {}
属性的装饰器
属性装饰器是指修饰类中声明的属性,用于在类中添加元数据、修改属性的行为。常见的属性装饰器有:
function myPropDecorator(target: any, key: string) {
// 修改属性的行为,添加元数据等
}
class MyClass {
// 使用属性装饰器
@myPropDecorator
myProp: number;
}
方法的装饰器
方法装饰器是指修饰类中声明的方法,用于在方法中添加元数据、修改方法行为。常见的方法装饰器有:
function myMethodDecorator(target: any, key: string, descriptor: PropertyDescriptor) {
// 修改方法的行为,添加元数据等
}
class MyClass {
// 使用方法装饰器
@myMethodDecorator
myMethod() {}
}
三、泛型编程的技巧
理解泛型中的类型约束
泛型能帮助我们在编写代码时写出更加通用的逻辑。但是泛型也存在着一些坑,例如类型约束的问题。当我们定义一个泛型函数时,需要确保输入参数类型的兼容性:
function fn<T extends { length: number }>(arg: T): T {
console.log(arg.length);
return arg;
}
fn(1); // Error,number 没有 length 属性
fn('hello'); // OK,string 有 length 属性
使用泛型编写可复用的代码
泛型还可以帮助我们编写更加通用的代码,提高代码的可复用性。例如常见的数组去重操作:
function unique<T>(arr: T[]): T[] {
return Array.from(new Set(arr));
}
const arr = [1, 2, 3, 3, 4, 4, 5];
const uniqueArr = unique(arr); // [1, 2, 3, 4, 5]
四、类型推导技巧
使用 never 类型提高代码健壮性
TypeScript 中有一个 never
类型,它表示的是那些永远不会出现的值的类型。在代码中,我们可以利用 never
类型来确保代码的健壮性。例如:
function throwError(msg: string): never {
throw new Error(msg);
}
function isNotNull<T>(val: T): val is NonNullable<T> {
if (val === null || val === undefined) {
throwError('value is null or undefined');
}
return true;
}
const str: string | null = null;
if (isNotNull(str)) {
// str 不可能是 null,这里会推导出类型为 string
console.log(str.length);
}
使用 keyof 获取对象属性名称
keyof
是 TypeScript 中的一种关键字,它可以获取某个对象属性的名称:
interface Person {
name: string;
age: number;
}
type PersonKey = keyof Person; // 'name' | 'age'
自定义索引类型,规范对象结构
在开发中,我们可能需要自定义一些索引类型来规范对象的结构:
interface Person {
name: string;
age: number;
}
type PartialPerson = Partial<Person>; // 把 Person 所有属性变成可选属性
type RequiredPerson = Required<Person>; // 把 Person 所有属性变成必选属性
type OmitPerson = Omit<Person, 'age'>; // 从 Person 中删除 age 属性
五、常用工具类型解析
Partial 类型,将类型属性变成可选的
Partial
是 TypeScript 中的一个工具类型,它可以把一个类型中所有属性变成可选的:
interface Person {
name: string;
age: number;
}
type PartialPerson = Partial<Person>; // 把 Person 所有属性变成可选属性
Required类型,将类型属性变成必选的
与 Partial
相反,Required
可以把一个类型中所有属性变成必选的:
interface Person {
name?: string;
age?: number;
}
type RequiredPerson = Required<Person>; // 把 Person 所有属性变成必选属性
Omit 类型,从类型中删除某个属性
Omit
可以从一个类型中删除指定的属性:
interface Person {
name: string;
age: number;
gender: string;
}
type WithoutGender = Omit<Person, 'gender'>; // 从 Person 中删除 gender 属性
六、高效应对循环和异步场景
使用 async/await 优雅处理异步操作
在处理异步操作时,我们通常会使用 Promise,而在 TypeScript 中,可以使用 async/await 更加优雅的处理异步操作。例如:
async function fetchUserInfo() {
const res = await fetch('https://api.com/userinfo');
const data = await res.json();
return data;
}
fetchUserInfo().then((userInfo) => {
console.log(userInfo);
});
数据处理循环的技巧
在数据处理中,我们经常使用循环来遍历数据。而在 TypeScript 中,可以使用 for...of
循环更加简化代码,同时提高代码的可读性和可维护性。例如:
interface Person {
name: string;
age: number;
}
const people: Person[] = [
{ name: 'Alice', age: 20 },
{ name: 'Bob', age: 25 },
{ name: 'Charlie', age: 30 },
];
for (const { name, age } of people) {
console.log(name, age);
}
总结
本文介绍了 TypeScript 中一些实用技巧,包括声明文件的使用技巧、使用装饰器提升代码可读性和可维护性、泛型编程技巧、类型推导技巧和常用工具类型解析、高效应对循环和异步场景等。希望这些技巧能帮助你写出更加优秀的 TypeScript 代码,成为真正的全栈工程师!
TypeScript实用技巧大杂烩,助你成为真正的全栈工程师的更多相关文章
- 【待考察】Appium使用技巧,助你快速入门移动端自动化!
Appium使用技巧,助你快速入门移动端自动化! 原创: 柠檬班superman 柠檬班软件测试 1月4日 关注并置顶[柠檬班]的小哥哥小姐姐 “猪”年行大运 说说最近研究移动端的自动化 移动端的自动 ...
- Notepad++ 实用技巧
Notepad++是一款开源的文本编辑器,功能强大.很适合用于编辑.注释代码.它支持绝大部分主流的编程语言. 本文主要列举了本人在实际使用中遇到的一些技巧. 快捷键 自定义快捷键 首先,需要知道的是: ...
- javascript实用技巧、javascript高级技巧
字号+作者:H5之家 来源:H5之家 2016-10-31 11:00 我要评论( ) 三零网提供网络编程. JavaScript 的技术文章javascript实用技巧.javascript高级技巧 ...
- iOS开发实用技巧—Objective-C中的各种遍历(迭代)方式
iOS开发实用技巧—Objective-C中的各种遍历(迭代)方式 说明: 1)该文简短介绍在iOS开发中遍历字典.数组和集合的几种常见方式. 2)该文对应的代码可以在下面的地址获得:https:// ...
- iOS开发实用技巧—在手机浏览器头部弹出app应用下载提示
iOS开发实用技巧—在手机浏览器头部弹出app应用下载提示 本文介绍其简单使用: 第一步:在本地建立一个访问的服务端. 打开本地终端,在本地新建一个文件夹,在该文件夹中存放测试的html页面. ...
- iOS开发实用技巧—项目新特性页面的处理
iOS开发实用技巧篇—项目新特性页面的处理 说明:本文主要说明在项目开发中会涉及到的最最简单的新特性界面(实用UIScrollView展示多张图片的轮播)的处理. 代码示例: 新建一个专门的处理新特性 ...
- IOS 网络浅析-(十三 SDWebImage 实用技巧)
IOS 网络浅析-(十三 SDWebImage 实用技巧) 首先让我描述一下为了什么而产生的实用技巧.(在TableView.CollectionView中)当用户所处环境WiFi网速不够快(不能立即 ...
- NSString的八条实用技巧
NSString的八条实用技巧 有一篇文章写了:iOS开发之NSString的几条实用技巧 , 今天这篇,我们讲讲NSString的八条实用技巧.大家可以收藏起来,方便开发随时可以复制粘贴. 0.首字 ...
- ###《VIM实用技巧》
###<VIM实用技巧> #@author: gr #@date: 2015-11-20 #@email: forgerui@gmail.com <VIM实用技巧>阅读笔记. ...
- PowerDesigner实用技巧小结(3)
PowerDesigner实用技巧小结(3) PowerDesigner 技巧小结 sqlserver数据库databasevbscriptsqldomain 1.PowerDesigner 使用 M ...
随机推荐
- vue+elementui怎样点击table中的单元格触发事件--弹框
可以先看一下官网中table的自定义列模板代码 <template> <el-table :data="tableData" border s ...
- Processing模拟控制多台舵机-以距离为参数 程序参考
又是一次课程学习的结束,辅导学生的过程也很受益,温故而知新.该组同学需要Arduino控制多达6个舵机,而且基于距离这一参数,在不同距离值之间会有不同的触发事件,也就是6个舵机转的角度都有所不同,而且 ...
- OpenOffice已停更六年 文档基金会呼吁用户迁移LibreOffice
OpenOffice 源代码发布至今已经有 20 年了.文档基金会近日发布公开信,表示说:"LibreOffice 是 OpenOffice 的未来.我们将会全力以赴".以此呼吁那 ...
- mysql : 第5章 数据库的安全性
-- 创建用户CREATE USER utest@localhost IDENTIFIED BY 'temp';-- 查看所有用户SELECT * FROM mysql.user;-- 查看表级权限S ...
- Java的数据类型详解
java的为强类型语言,所以要求变量的使用要严格符合规定,所有的变量都必须先定义后在使用: 什么是变量? 变量顾名思义,就是可变的量:是程序中最基本的存储单元,其要素要包括:变量名.变量类型和作用域: ...
- nginx 配置go服务反向代理
nginx 配置 详细请看Nginx 极简教程 server { listen 80; server_name localhost; #charset koi8-r; # nginx访问活动日志 ac ...
- go errors转string
前言 在 Go 中如果声明了两个字符相同的错误,但命名是新的变量,此时两个错误不相等 package main import ( "errors" "fmt" ...
- IM服务器:一个使用imserver服务器进行聊天的web端案例
该案例中包含一个基于web网页的前端程序,该案例会使用websocket与IM服务器(imserver)进行通信. 一.环境准备 1.下载 "imserver网页调用案例",并解压 ...
- weblogic日志查看
都是使用tail -f catalina.out命令来查看的,不要太顺手: 今天登陆到公司另外一个Team 的服务器上,发现装的是weblogic,手痒想看看weblogic怎么看日志,找了老半天,才 ...
- VS Code Runner 插件配置
VS Code Runner 插件配置 Code Runner插件是一个小而美的插件,可以很方便的运行一些简单的代码文件. 本篇博文记录一些相关的环境配置. 设置C++编译标准 这里可以设置默认的C+ ...