ts语法概览

ts对应js的数据类型

let name: string = "bob";
let age: number = 37;
let isDone: boolean = false;
let u: undefined = undefined;
let n: null = null;
let obj: object = {x: 1};
let bigLiteral: bigint = 100n;
let sym: symbol = Symbol("me");

Array

// 元素类型[]
let list: number[] = [1, 2, 3];
// Array<元素类型>
let list: Array<number> = [1, 2, 3];

interface MyObject {
   name: string;
   age: number;
}

let arr: MyObject[] = [{name: "兔兔", age: 18}] // OK
复制代码

Object

object表示非原始类型,也就是除number,string,boolean,symbol,null或undefined之外的类型。

declare function create(o: object | null): void;

Tuple

  • 对于内部不同类型的数组可以使用元组类型来定义
  • 注意,元组类型只能表示一个已知元素数量和类型的数组,长度已指定,越界访问会提示错误。例如,一个数组中可能有多种类型,数量和类型都不确定,那就直接any[]。
let x: [string, number];

x = ['hello', 10]; // OK
x = [10, 'hello']; // Error

undefined和null

  • 默认情况下null和undefined是所有类型的子类型。 就是说你可以把null和undefined赋值给任何类型的变量。

void

  • void表示没有任何类型,和其他类型是平等关系,不能直接赋值
  • 一般也只有在函数没有返回值时去声明

any和unknown

  • any(既是top type又是bottom type)会跳过类型检查器对值的检查,任何类型的值可以赋值给any,同时any类型的值也可以赋值给任何类型
  • 所有类型都可以分配给unknown(top type),但它只能赋值给unknown和any

never

  • never类型表示的是那些永不存在的值的类型。
  • 比如函数执行时抛出异常或者函数中的死循环,不存在返回值
  • 是任何类型的子类型,也可以赋值给任何类型,但没有类型可以赋值给never类型

类型断言

  • 类型转换通常发生在你比TS更了解某个值的详细信息的时候。
    其一是“尖括号”语法
let someValue: any = "this is a string";
let strLength: number = (<string>someValue).length;

另一个是as 语法

let someValue: any = "this is a string";
let strLength: number = (someValue as string).length;

!

跳过ts校验,保证有值

// 断言event.target有值,不为null/undefined
const a:string = event.target!.name

类型推论

  • 如果没有明确的指定类型,那么 TypeScript 会依照类型推论的规则推断出一个类型。
let myFavoriteNumber = 'seven';
myFavoriteNumber = 7; // Error

| 联合类型

  • 联合类型表示取值可以为多种类型中的一种,使用 | 分隔每个类型
let myFavoriteNumber: string | number;
myFavoriteNumber = 'seven'; // OK
myFavoriteNumber = 7; // OK

& 交叉类型

  • 交叉类型是将多个类型合并为一个类型
  • 如果key相同但是类型不同,则该key为never类型。
interface A {
  name: string,
  age: number
}
interface B {
  name: string,
  gender: string
}

let a: A & B = { // OK
    name: "兔兔",
    age: 18,
    gender: "男"
};

interface 接口

  • 定义一些参数,规定变量里面有什么参数,参数是什么类型,使用时就必须有这些对应类型的参数,少或者多参数、参数类型不对都会报错
interface LabeledValue {
  label: string;
  money?: number; // 可选属性
  readonly x: number; // 只读属性
  [key: string]: any; // 索引签名,TypeScript 支持两种索引签名:字符串和数字。
}
function printLabel(labeledObj: LabeledValue) {
  console.log(labeledObj.label);
}
// myObj中的属性会进行类型推论,以此绕过多余的类型检查
let myObj = { size: 10, label: "Size 10 Object" };
printLabel(myObj); // OK

ReadonlyArray

  • 数组创建后再也不能被修改
let a: number[] = [1, 2, 3, 4];
let ro: ReadonlyArray<number> = a;
ro[0] = 12; // Error
ro.push(5); // Error
ro.length = 100; // Error

a = ro; // Error
a = ro as number[]; // OK

extends 接口继承接口

  • 接口可以多继承。
  • 使用多继承时,先确保父接口没有共有属性,或共有属性定义的类型都相同。
interface Shape {
  color: string;
}
interface PenStroke {
  penWidth: number;
}
interface Square extends Shape, PenStroke {
  sideLength: number;
}

let square: Square = { sideLength: 1 } // Error
let square1: Square = { sideLength: 1, color: 'red' } // Error
let square2: Square = { sideLength: 1, color: 'red', penWidth: 2 } // OK

Pick 选择性继承

  • 主要是从一个已知的类型中,取出子集,作为一个新的类型返回
// 原始类型
interface TState {
    name: string;
    age: number;
    like: string[];
}
// 如果我只想要name和age
interface TSingleState extends Pick<TState, "name" | "age"> {};
或者
interface TSingleState extends Pick<TState, "name" | "age"> ;

函数声明

// 剩余参数
function sum(x: number, y: number, ...rest: any[]): number {
    return x + y;
}

函数表达式

// 可选参数、参数默认值
let mySum: (x: number, y: number, m:number = 11, z?: number) => number = function (x: number, y: number): number {
    return x + y;
};

用接口定义函数类型

interface SearchFunc{
  (source: string, subString: string): boolean;
}

let mySearch: SearchFunc = function(source: string, subString: string) { // OK
  let result = source.search(subString);
  return result >-1;
};

内置对象

let s: String = new String('兔神');
let n: Number = new Number(123);
let b: Boolean = new Boolean(1);
let e: Error = new Error('Error occurred');
let d: Date = new Date();
let r: RegExp = /[a-z]/;

let body: HTMLElement = document.body;
let allDiv: NodeList = document.querySelectorAll('div');
document.addEventListener('click', function(e: MouseEvent) {
  // Do something
});

function sum() {
    let args: IArguments = arguments; // 类数组对象
}

type 类型别名

  • 类型别名就是给一种类型起个别的名字,使用type关键字来定义
type StringType = string;
let str: StringType;
str = 'hello';
str = 123 // Error

import type ... from & export type ... from

导入或者导出类型文件,编译器将把这些类型文件删除

  • import type ... from — 让编译器知道您要导入的内容绝对是一种类型。
  • export type ... from — 一样, 仅用作导出。

// src/lib-type-re-export.ts
export type { Track, Playlist } from "./types"; // 类型文件
export type { CreatePlaylistRequestParams } from "./api"; // 类型文件
export { createPlaylist } from "./api";

// 会被编译为:

// dist/lib-type-re-export.js
export { createPlaylist } from "./api";

字符串字面量类型

  • 字符串字面量类型用来约束取值只能是某几个字符串中的一个。
type Name = 'ALisa' | 'Bob' | 'Cola'

let name: Name = 'Alisa'; // Error 与 DOM 中的全局 window 对象下的 name 属性出现了重名
let name1: Name = 'ALisa'; // OK
let name2: Name = 'Bob'; // OK
let name3: Name = 'Cola'; // OK
let name4: Name = '兔兔'; // Error

Enum 枚举

  • 枚举是一个被命名的整型常数的集合
  • TS支持数字的和基于字符串的枚举。
// 数字枚举
enum Days {Sun, Mon, Tue, Wed, Thu, Fri, Sat};
console.log(Days.Sun) // 0
console.log(Days.Mon) // 1
// 初始化枚举成员,那么该初始化成员后面的成员会在它的基础上自动增长1
enum Days {Sun = 1, Mon, Tue, Wed, Thu, Fri, Sat};
console.log(Days.Sun) // 1
console.log(Days.Mon) // 2

// 字符串枚举
enum Direction {
    Up = "UP",
    Down = "DOWN",
    Left = "LEFT",
    Right = "RIGHT",
}
// 异构枚举,枚举可以混合字符串和数字成员
enum Direction {
    name = '兔兔',
    age = 18
}

class 类

class Animal {
    // public 修饰符,类的属性、方法可以在外部访问
    public static age: number = 18;
    // 类的属性、方法不可以在外部访问
    private static title: string = '兔兔';
    // protected 修饰符,在派生类中仍然可以访问(继承中可以访问)
    protected title: string = '兔兔';
    class Animal {
    constructor(public name: string, private age: number, protected sex: string) {
    
    }
}

Animal.age; // OK
Animal.title; // Error

abstract 抽象类

  • 象类做为其它派生类的基类使用, 不允许被实例化。
  • 抽象类中的抽象方法不包含具体实现并且必须在派生类中实现
abstract class Department {
    constructor(public name: string) {
    }
    printName(): void {
        console.log('Department name: ' + this.name);
    }
    abstract printMeeting(): void; // 必须在派生类中实现
}
class AccountingDepartment extends Department {
    constructor() {
        super('Accounting and Auditing'); // 在派生类的构造函数中必须调用 super()
    }
    printMeeting(): void {
        console.log('The Accounting Department meets each Monday at 10am.');
    }
    generateReports(): void {
        console.log('Generating accounting reports...');
    }
}
let department: Department; // OK:允许创建一个对抽象类型的引用
department = new Department(); // Error: 不能创建一个抽象类的实例
department = new AccountingDepartment(); // OK:允许对一个抽象子类进行实例化和赋值
department.printName(); // OK
department.printMeeting(); // OK
department.generateReports(); // Error: 方法在声明的抽象类中不存在
复制代码

implements 类实现接口

  • 用类去实现接口
interface Age {
  age: number;
}

interface Title{
  title: string;
}
class title implements Title, Age{
  title: string = '兔兔';
  age: number = 18;
}

T 泛型

  • 泛型是指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定类型的一种特性。
// 泛型函数
function identity<T>(arg: T): T {
    return arg;
}
identity<number>(1); // OK:明确的指定`T`是`number`类型
identity(1); // OK:让编译器自己推断类型

// 泛型约束,在函数内部使用泛型变量的时候,由于事先不知道它是哪种类型,所以不能随意的操作它的属性或方法。
function loggingIdentity<T>(arg: T): T {
    console.log(arg.length); // Error
    return arg;
}
function loggingIdentity<T>(arg: T[]): T[] {
    console.log(arg.length);  // OK
    return arg;
}

// 泛型接口
interface Person<T> {
    name: T;
    getAge(arg: T): T;
}

let myIdentity: Person<string> = {
    name: "兔兔",
    getAge(name) {
        return name
    }
};

// 泛型类
class GenericNumber<T> {
    zeroValue: T;
    add: (x: T, y: T) => T;
}
let myGenericNumber = new GenericNumber<number>();

// 泛型参数的默认类型
function createArray<T = string>(length: number, value: T): Array<T> {
}

类型守卫

// 要定义一个类型守卫,我们只要简单地定义一个函数,它的返回值是一个类型谓词
function isA(person: A | B): person is A{
    return(person as A).name !== undefined;
}

// 使用
function doSomething(person: A | B): void {
    if(isA(person)) { // OK
        // ...
    }
}

// 使用in操作符
// typeof类型守卫
// instanceof类型守卫
// 
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • 以自己作为客户端RD的视角,列举了一些在实际使用中比较常见,而又比较陌生的语法(主要集中在ES6的特性上),做好一...
    孢子菌阅读 1,487评论 0 2
  • 本文列举了一些在实际使用中比较常见,而又比较陌生的语法(主要集中在ES6的特性上),做好一些铺垫和基础。JavaS...
    平凡的雪夜冬天阅读 764评论 0 0
  • Typescript 基本语法 概念 TypeScript 是一种强类型语言 什么是强类型语言 不允许改变变量的数...
    抽疯的稻草绳阅读 10,850评论 0 7
  • TypeScript是微软开发的,基于类的面向对象编程,其文件以 .ts 为后缀名; TypeScript是Jav...
    hellomyshadow阅读 955评论 0 0
  • TS 入门和Vue实践 一、TS 快速上手 从 JavaScript 程序员的角度总结思考,快速上手理解 Type...
    以虚名称之阅读 21,190评论 0 15