• TypeScript 快速入门之语法详解(一篇文章精通系列)【webpack】(常用语法【二】)【WebStorm版本】


    1、基础类型

    紧接上篇文章建立的项目(项目案例模板下载:https://download.csdn.net/download/qq_44757034/86504095
    在这里插入图片描述
    在这里插入图片描述
    在main.ts当中引入
    在这里插入图片描述

    import './01_基本数据类型'
    
    
    • 1
    • 2

    在01_基本数据类型.ts当中写一些内容

    在这里插入图片描述
    运行项目
    在这里插入图片描述
    在这里插入图片描述

    (1)布尔值

    最基本的数据类型就是简单的 true/false 值,在JavaScript 和 TypeScript 里叫做 boolean(其它语言中也一样)。
    在这里插入图片描述

    let isDone: boolean = false;
    isDone = true;
    // isDone = 2 // error
    
    • 1
    • 2
    • 3

    运行测试
    在这里插入图片描述

    (2)数字

    和 JavaScript 一样,TypeScript 里的所有数字都是浮点数。 这些浮点数的类型是 number。 除了支持十进制和十六进制字面量,TypeScript 还支持 ECMAScript 2015中引入的二进制和八进制字面量。
    在这里插入图片描述

    let a1: number = 10 // 十进制
    let a2: number = 0b1010  // 二进制
    let a3: number = 0o12 // 八进制
    let a4: number = 0xa // 十六进制
    
    • 1
    • 2
    • 3
    • 4

    在这里插入图片描述

    (3)字符串

    JavaScript 程序的另一项基本操作是处理网页或服务器端的文本数据。 像其它语言里一样,我们使用 string 表示文本数据类型。 和 JavaScript 一样,可以使用双引号(")或单引号(')表示字符串。
    在这里插入图片描述

    
    let username:string = 'tom'
    // @ts-ignore
    username = 'jack'
    // name = 12 // error
    let age:number = 12
    const info = `My name is ${username}, I am ${age} years old!`
    console.log(info)
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    在这里插入图片描述

    (4)undefined 和 null

    TypeScript 里,undefined 和 null 两者各自有自己的类型分别叫做 undefined 和 null。 它们的本身的类型用处不是很大:
    在这里插入图片描述

    let u: undefined = undefined
    let n: null = null
    
    • 1
    • 2

    默认情况下 null 和 undefined 是所有类型的子类型。 就是说你可以把 null 和 undefined 赋值给 number 类型的变量。
    在这里插入图片描述

    (5)数组

    TypeScript 像 JavaScript 一样可以操作数组元素。 有两种方式可以定义数组。 第一种,可以在元素类型后面接上[],表示由此类型元素组成的一个数组:

    let list1: number[] = [1, 2, 3]
    
    • 1

    第二种方式是使用数组泛型,Array<元素类型>:

    let list2: Array<number> = [1, 2, 3]
    
    • 1

    在这里插入图片描述

    let list1: number[] = [1, 2, 3]
    let list2: Array<number> = [1, 2, 3]
    
    console.log(list1)
    console.log(list2)
    
    list1.forEach((value, index, array)=>{
        console.log(value + ":"+index+":"+array[index])
    })
    list2.forEach((value, index, array)=>{
        console.log(value + ":"+index+":"+array[index])
    })
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    在这里插入图片描述

    (6)元组 Tuple

    元组类型允许表示一个已知元素数量和类型的数组,各元素的类型不必相同。
    比如,你可以定义一对值分别为 string 和 number 类型的元组。
    在这里插入图片描述

    let t1: [string, number]
    t1 = ['hello', 10] // OK
    t1 = [10, 'hello'] // Error
    
    • 1
    • 2
    • 3

    循环遍历
    在这里插入图片描述

    t1.forEach((value)=>{
        console.log(value)
    })
    
    • 1
    • 2
    • 3

    当访问一个已知索引的元素,会得到正确的类型:

    console.log(t1[0].substring(1)) // OK
    console.log(t1[1].substring(1)) // Error, 'number' 不存在 'substring' 方法
    
    • 1
    • 2

    (7)枚举

    enum 类型是对 JavaScript 标准数据类型的一个补充。 使用枚举类型可以为一组数值赋予友好的名字。
    在这里插入图片描述

    enum Color {
      Red,
      Green,
      Blue
    }
    
    // 枚举数值默认从0开始依次递增
    // 根据特定的名称得到对应的枚举数值
    let myColor: Color = Color.Green  // 1
    console.log(myColor, Color.Red, Color.Blue)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    在这里插入图片描述
    默认情况下,从 0 开始为元素编号。

    你也可以手动的指定成员的数值。

    例如,我们将上面的例子改成从 1 开始编号:
    在这里插入图片描述

    enum Color {Red = 1, Green, Blue}
    let c: Color = Color.Green
    
    • 1
    • 2

    在这里插入图片描述
    或者,全部都采用手动赋值:
    在这里插入图片描述

    enum Color {Red = 1, Green = 2, Blue = 4}
    let c: Color = Color.Green
    
    • 1
    • 2
    enum Color {Red = 1, Green = 2, Blue = 4}
    let c: Color = Color.Blue
    
    console.log(c)
    
    • 1
    • 2
    • 3
    • 4

    枚举类型提供的一个便利是你可以由枚举的值得到它的名字。 例如,我们知道数值为 2,但是不确定它映射到 Color 里的哪个名字,我们可以查找相应的名字:

    enum Color {Red = 1, Green, Blue}
    let colorName: string = Color[2]
    
    console.log(colorName)  // 'Green'
    
    • 1
    • 2
    • 3
    • 4

    在这里插入图片描述

    enum Color {Red = 1,Green,Blue}
    
    let colorName:string = Color[2]
    
    console.log(colorName)
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    在这里插入图片描述

    (8)any

    有时候,我们会想要为那些在编程阶段还不清楚类型的变量指定一个类型。

    这些值可能来自于动态的内容,比如来自用户输入或第三方代码库。

    这种情况下,我们不希望类型检查器对这些值进行检查而是直接让它们通过编译阶段的检查。

    那么我们可以使用 any 类型来标记这些变量:

    let notSure: any = 4
    notSure = 'maybe a string'
    notSure = false // 也可以是个 boolean
    
    • 1
    • 2
    • 3

    在这里插入图片描述
    在这里插入图片描述
    在对现有代码进行改写的时候,any 类型是十分有用的,

    它允许你在编译时可选择地包含或移除类型检查。

    并且当你只知道一部分数据的类型时,any 类型也是有用的。

    比如,你有一个数组,它包含了不同的类型的数据:

    let list: any[] = [1, true, 'free']
    
    list[1] = 100
    
    • 1
    • 2
    • 3

    在这里插入图片描述
    在这里插入图片描述

    (9)void

    某种程度上来说,void 类型像是与 any 类型相反,它表示没有任何类型。

    当一个函数没有返回值时,你通常会见到其返回值类型是 void:

    /* 表示没有任何类型, 一般用来说明函数的返回值不能是undefined和null之外的值 */
    function fn(): void {
      console.log('fn()')
      // return undefined
      // return null
      // return 1 // error
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    在这里插入图片描述
    运行测试
    在这里插入图片描述
    声明一个 void 类型的变量没有什么大用,因为你只能为它赋予 undefined 和 null:

    let unusable: void = undefined
    
    • 1

    在这里插入图片描述
    在这里插入图片描述

    (10)object

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

    使用 object 类型,就可以更好的表示像 Object.create 这样的 API。

    例如:
    在这里插入图片描述

    
    function fn2(obj:object):object {
        console.log('fn2()', obj)
        return {}
        // return undefined
        // return null
    }
    console.log(fn2(new String('abc')))
    // console.log(fn2('abc') // error
    console.log(fn2(String))
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    在这里插入图片描述

    (11)联合类型

    联合类型(Union Types)表示取值可以为多种类型中的一种
    需求1: 定义一个一个函数得到一个数字或字符串值的字符串形式值

    在这里插入图片描述

    function toString2(x: number | string) : string {
        return x.toString()
    }
    
    console.log(toString2(1))
    console.log(toString2('1'))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    在这里插入图片描述
    需求2: 定义一个一个函数得到一个数字或字符串值的长度

    function getLength(x: number | string) {
    
      // return x.length // error
    
      if (x.length) { // error
        return x.length
      } else {
        return x.toString().length
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    (12)类型断言

    通过类型断言这种方式可以告诉编译器,“相信我,我知道自己在干什么”。
    类型断言好比其它语言里的类型转换,但是不进行特殊的数据检查和解构。
    它没有运行时的影响,只是在编译阶段起作用。
    TypeScript 会假设你,程序员,已经进行了必须的检查。

    类型断言有两种形式。
    其一是“尖括号”语法, 另一个为 as 语法
    在参数多种类型的时候,当传入数据不确定的时候,需要通过断言确定对应的数据类型

    /* 
    类型断言(Type Assertion): 可以用来手动指定一个值的类型
    语法:
        方式一: <类型>值
        方式二: 值 as 类型  tsx中只能用这种方式
    */
    
    /* 需求: 定义一个函数得到一个字符串或者数值数据的长度 */
    function getLength(x: number | string) {
      if ((<string>x).length) {
        return (x as string).length
      } else {
        return x.toString().length
      }
    }
    console.log(getLength('abcd'), getLength(1234))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    在这里插入图片描述

    (13)类型推断

    类型推断: TS会在没有明确的指定类型的时候推测出一个类型
    有下面2种情况:

    1. 定义变量时赋值了, 推断为对应的类型.
    2. 定义变量时没有赋值, 推断为any类型
    /* 定义变量时赋值了, 推断为对应的类型 */
    let b9 = 123 // number
    // b9 = 'abc' // error
    
    /* 定义变量时没有赋值, 推断为any类型 */
    let b10  // any类型
    b10 = 123
    b10 = 'abc'
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    在这里插入图片描述
    在这里插入图片描述

    2、接口

    TypeScript 的核心原则之一是对值所具有的结构进行类型检查。我们使用接口(Interfaces)来定义对象的类型。接口是对象的状态(属性)和行为(方法)的抽象(描述)
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

    (1)创建接口

    需求: 创建人的对象, 需要对人的属性进行一定的约束

    id是number类型, 必须有, 只读的
    name是string类型, 必须有
    age是number类型, 必须有
    sex是string类型, 可以没有
    
    • 1
    • 2
    • 3
    • 4

    下面通过一个简单示例来观察接口是如何工作的:

    /* 
    在 TypeScript 中,我们使用接口(Interfaces)来定义对象的类型
    接口: 是对象的状态(属性)和行为(方法)的抽象(描述)
    接口类型的对象
        多了或者少了属性是不允许的
        可选属性: ?
        只读属性: readonly
    */
    /* 
    需求: 创建人的对象, 需要对人的属性进行一定的约束
      id是number类型, 必须有, 只读的
      name是string类型, 必须有
      age是number类型, 必须有
      sex是string类型, 可以没有
    */
    // 定义人的接口
    interface IPerson {
      id: number
      name: string
      age: number
      sex: string
    }
    const person1: IPerson = {
      id: 1,
      name: 'tom',
      age: 20,
      sex: '男'
    }
    console.log(person1)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29

    在这里插入图片描述
    在这里插入图片描述
    类型检查器会查看对象内部的属性是否与IPerson接口描述一致, 如果不一致就会提示类型错误

    (2)可选属性

    接口里的属性不全都是必需的。 有些是只在某些条件下存在,或者根本不存在

    interface IPerson {
      id: number
      name: string
      age: number
      sex?: string
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    带有可选属性的接口与普通的接口定义差不多,只是在可选属性名字定义的后面加一个 ? 符号。
    可选属性的好处之一是可以对可能存在的属性进行预定义,好处之二是可以捕获引用了不存在的属性时的错误。

    const person2: IPerson = {
      id: 1,
      name: 'tom',
      age: 20,
      // sex: '男' // 可以没有
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    在这里插入图片描述
    在这里插入图片描述
    如果不加问号会报错
    在这里插入图片描述

    (3)只读属性

    一些对象属性只能在对象刚刚创建的时候修改其值。 你可以在属性名前用 readonly 来指定只读属性:

    interface IPerson {
      readonly id: number
      name: string
      age: number
      sex?: string
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    一旦赋值后再也不能被改变了。

    const person2: IPerson = {
      id: 2,
      name: 'tom',
      age: 20,
      // sex: '男' // 可以没有
      // xxx: 12 // error 没有在接口中定义, 不能有
    }
    person2.id = 2 // error
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    readonly vs const

    最简单判断该用 readonly 还是 const 的方法是看要把它做为变量使用还是做为一个属性。

    做为变量使用的话用 const,若做为属性则使用 readonly。
    在这里插入图片描述
    在这里插入图片描述

    (4)函数类型

    接口能够描述 JavaScript 中对象拥有的各种各样的外形。

    除了描述带有属性的普通对象外,接口也可以描述函数类型。

    为了使用接口表示函数类型,我们需要给接口定义一个调用签名。

    它就像是一个只有参数列表和返回值类型的函数定义。

    参数列表里的每个参数都需要名字和类型。

    /* 
    接口可以描述函数类型(参数的类型与返回的类型)
    */
    interface SearchFunc {
    	//参数,返回值类型
      (source: string, subString: string): boolean
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    这样定义后,我们可以像使用其它接口一样使用这个函数类型的接口。
    下例展示了如何创建一个函数类型的变量,并将一个同类型的函数赋值给这个变量。

    const mySearch: SearchFunc = function (source: string, sub: string): boolean {
      return source.search(sub) > -1
    }
    console.log(mySearch('abcd', 'bc'))
    
    • 1
    • 2
    • 3
    • 4

    在这里插入图片描述
    在这里插入图片描述

    (5)类类型

    类实现接口

    与 C# 或 Java 里接口的基本作用一样,

    TypeScript 也能够用它来明确的强制一个类去符合某种契约。

    /* 
    类类型: 实现接口
    1. 一个类可以实现多个接口
    2. 一个接口可以继承多个接口
    */
    interface Alarm {
      alert(): any;
    }
    interface Light {
      lightOn(): void;
      lightOff(): void;
    }
    class Car implements Alarm {
      alert() {
          console.log('Car alert');
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    在这里插入图片描述
    在这里插入图片描述

    (6)一个类可以实现多个接口

    在这里插入图片描述

    class Car2 implements Alarm, Light {
        alert() {
            console.log('Car alert');
        }
        lightOn() {
            console.log('Car light on');
        }
        lightOff() {
            console.log('Car light off');
        }
    }
    var car2 = new Car2();
    car2.alert()
    car2.lightOn()
    car2.lightOff()
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    在这里插入图片描述

    (7)接口继承接口

    和类一样,接口也可以相互继承。

    这让我们能够从一个接口里复制成员到另一个接口里,可以更灵活地将接口分割到可重用的模块里。

    interface LightableAlarm extends Alarm, Light {
    }
    
    • 1
    • 2

    3、类

    对于传统的 JavaScript 程序我们会使用函数和基于原型的继承来创建可重用的组件,但对于熟悉使用面向对象方式的程序员使用这些语法就有些棘手,因为他们用的是基于类的继承并且对象是由类构建出来的。

    从 ECMAScript 2015,也就是 ES6 开始, JavaScript 程序员将能够使用基于类的面向对象的方式。

    使用 TypeScript,我们允许开发者现在就使用这些特性,并且编译后的 JavaScript 可以在所有主流浏览器和平台上运行,而不需要等到下个 JavaScript 版本。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

    (1)基本示例

    下面看一个使用类的例子:
    在这里插入图片描述

    /* 
    类的基本定义与使用
    */
    
    class Greeter {
      // 声明属性
      message: string
    
      // 构造方法
      constructor (message: string) {
        this.message = message
      }
    
      // 一般方法
      greet (): string {
        return 'Hello ' + this.message
      }
    }
    
    // 创建类的实例
    const greeter = new Greeter('world')
    // 调用实例的方法
    console.log(greeter.greet())
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    如果你使用过 C# 或 Java,你会对这种语法非常熟悉。 我们声明一个 Greeter 类。

    这个类有 3 个成员:一个叫做 message 的属性,一个构造函数和一个 greet 方法。

    你会注意到,我们在引用任何一个类成员的时候都用了 this。 它表示我们访问的是类的成员。

    后面一行,我们使用 new 构造了 Greeter 类的一个实例。

    它会调用之前定义的构造函数,创建一个 Greeter 类型的新对象,并执行构造函数初始化它。

    最后一行通过 greeter 对象调用其 greet 方法
    在这里插入图片描述

    (2)继承

    在 TypeScript 里,我们可以使用常用的面向对象模式。

    基于类的程序设计中一种最基本的模式是允许使用继承来扩展现有的类。

    看下面的例子
    在这里插入图片描述

    class Animal{
        run(distance:number){
            console.log(`Animal run ${distance}m`)
        }
    }
    class Dog extends Animal{
        cry(){
            console.log('wang! wang!')
        }
    }
    const dog = new Dog();
    
    dog.cry()
    dog.run(100)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    在这里插入图片描述
    这个例子展示了最基本的继承:类从基类中继承了属性和方法。 这里,Dog 是一个 派生类,它派生自 Animal 基类,通过 extends 关键字。 派生类通常被称作子类,基类通常被称作超类。

    因为 Dog 继承了 Animal 的功能,因此我们可以创建一个 Dog 的实例,它能够 cry() 和 run()。

    下面我们来看个更加复杂的例子。

    (3)继承

    在这里插入图片描述

    
    class Horse extends Animal {
        constructor (name: string) {
            // 调用父类型构造方法
            super(name)
        }
        // 重写父类型的方法
        run (distance: number=50) {
            console.log('dashing...')
            // 调用父类型的一般方法
            super.run(distance)
        }
        xxx () {
            console.log('xxx()')
        }
    }
    const snake = new Snake('sn')
    snake.run()
    
    const horse = new Horse('ho')
    horse.run()
    
    // 父类型引用指向子类型的实例 ==> 多态
    const tom: Animal = new Horse('ho22')
    tom.run()
    
    /* 如果子类型没有扩展的方法, 可以让子类型引用指向父类型的实例 */
    const tom3: Snake = new Animal('tom3')
    tom3.run()
    /* 如果子类型有扩展的方法, 不能让子类型引用指向父类型的实例 */
    const tom2: Horse = new Animal('tom2')
    tom2.run()
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33

    在这里插入图片描述
    这个例子展示了一些上面没有提到的特性。 这一次,我们使用 extends 关键字创建了 Animal的两个子类:Horse 和 Snake。

    与前一个例子的不同点是,派生类包含了一个构造函数,它 必须调用 super(),它会执行基类的构造函数。 而且,在构造函数里访问 this 的属性之前,我们 一定要调用 super()。 这个是 TypeScript 强制执行的一条重要规则。

    这个例子演示了如何在子类里可以重写父类的方法。Snake类和 Horse 类都创建了 run 方法,它们重写了从 Animal 继承来的 run 方法,使得 run 方法根据不同的类而具有不同的功能。注意,即使 tom 被声明为 Animal 类型,但因为它的值是 Horse,调用 tom.run(34) 时,它会调用 Horse 里重写的方法。

    sliding...
    sn run 5m
    dashing...
    ho run 50m
    
    • 1
    • 2
    • 3
    • 4

    4、公共,私有与受保护的修饰符

    • 默认为 public

    在上面的例子里,我们可以自由的访问程序里定义的成员。

    如果你对其它语言中的类比较了解,就会注意到我们在之前的代码里并没有使用 public 来做修饰;

    例如,C# 要求必须明确地使用 public 指定成员是可见的。

    在 TypeScript 里,成员都默认为 public。

    你也可以明确的将一个成员标记成 public。

    我们可以用下面的方式来重写上面的 Animal 类:

    • 理解 private

    当成员被标记成 private 时,它就不能在声明它的类的外部访问。

    • 理解 protected

    protected 修饰符与 private 修饰符的行为很相似,但有一点不同,
    protected成员在派生类中仍然可以访问。

    例如:
    在这里插入图片描述

    class Animal{
        public name:string
    
        constructor(name:string) {
            this.name = name
        }
        public run(distance:number = 0){
            console.log(`${this.name} run ${distance}m`)
        }
    }
    class Person extends Animal{
        private age:number = 18
        protected  sex:string = '男'
    
        run(distance:number=5){
            console.log('Person jumpng....')
            super.run(distance)
        }
    }
    class Student extends Person{
        run(distance:number = 5){
            console.log('Student jumping...')
            console.log(this.sex) // 子类能看到父类中受保护的成员
            // console.log(this.age) //  子类看不到父类中私有的成员
            super.run(distance)
        }
    }
    console.log(new Person('abc').name)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28

    在这里插入图片描述

    5、readonly 修饰符

    你可以使用 readonly 关键字将属性设置为只读的。

    只读属性必须在声明时或构造函数里被初始化。

    class Person {
      readonly name: string = 'abc'
      constructor(name: string) {
        this.name = name
      }
    }
    
    let john = new Person('John')
    // john.name = 'peter' // error
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    参数属性

    在上面的例子中,我们必须在 Person 类里定义一个只读成员 name 和一个参数为 name 的构造函数,

    并且立刻将 name 的值赋给 this.name,这种情况经常会遇到。

    参数属性可以方便地让我们在一个地方定义并初始化一个成员。

    下面的例子是对之前 Person 类的修改版,使用了参数属性:

    class Person2 {
      constructor(readonly name: string) {
      }
    }
    
    const p = new Person2('jack')
    console.log(p.name)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    注意看我们是如何舍弃参数 name,

    仅在构造函数里使用 readonly name: string 参数来创建和初始化 name 成员。

    我们把声明和赋值合并至一处。

    参数属性通过给构造函数参数前面添加一个访问限定符来声明。

    使用 private 限定一个参数属性会声明并初始化一个私有成员;

    对于 public 和 protected 来说也是一样。

    6、存取器

    TypeScript 支持通过 getters/setters 来截取对对象成员的访问。

    它能帮助你有效的控制对对象成员的访问。

    下面来看如何把一个简单的类改写成使用 get 和 set。

    首先,我们从一个没有使用存取器的例子开始。

    class Person {
      firstName: string = 'A'
      lastName: string = 'B'
      get fullName () {
        return this.firstName + '-' + this.lastName
      }
      set fullName (value) {
        const names = value.split('-')
        this.firstName = names[0]
        this.lastName = names[1]
      }
    }
    
    const p = new Person()
    console.log(p.fullName)
    
    p.firstName = 'C'
    p.lastName =  'D'
    console.log(p.fullName)
    
    p.fullName = 'E-F'
    console.log(p.firstName, p.lastName)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    7、存取器

    TypeScript 支持通过 getters/setters 来截取对对象成员的访问。

    它能帮助你有效的控制对对象成员的访问。

    下面来看如何把一个简单的类改写成使用 get 和 set。

    首先,我们从一个没有使用存取器的例子开始。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

    class Person1{
        firstName:string = 'A'
        lastName:string = 'B'
        get fullName(){
            return this.firstName + '-' +this.lastName
        }
        set fullName(value){
            const names = value.split('-');
            this.firstName = names[0]
            this.lastName = names[1]
        }
    }
    const p = new Person1();
    
    console.log(p.fullName)
    
    p.firstName = 'C'
    p.lastName = 'D'
    
    console.log(p.fullName)
    
    p.fullName = 'E-F'
    console.log(p.firstName,p.lastName)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    在这里插入图片描述

    8、静态属性

    到目前为止,我们只讨论了类的实例成员,那些仅当类被实例化的时候才会被初始化的属性。 我们也可以创建类的静态成员,这些属性存在于类本身上面而不是类的实例上。 在这个例子里,我们使用 static 定义 origin,因为它是所有网格都会用到的属性。 每个实例想要访问这个属性的时候,都要在 origin 前面加上类名。 如同在实例属性上使用 this.xxx 来访问属性一样,这里我们使用 Grid.xxx 来访问静态属性。
    在这里插入图片描述

    class Person2{
        name1:string = 'A'
        static name2:string = 'B'
    }
    console.log(Person2.name2)
    console.log(new Person2().name1)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    在这里插入图片描述

    9、抽象类

    抽象类做为其它派生类的基类使用。

    它们不能被实例化。

    不同于接口,抽象类可以包含成员的实现细节。

    abstract 关键字是用于定义抽象类和在抽象类内部定义抽象方法。
    在这里插入图片描述

    /*
    抽象类
      不能创建实例对象, 只有实现类才能创建实例
      可以包含未实现的抽象方法
    */
    abstract class Animal2 {
    
        // @ts-ignore
        abstract cry2 ()
    
        run () {
            console.log('run()')
        }
    }
    
    class Dog2 extends Animal2 {
        cry2 () {
            console.log(' Dog cry()')
        }
    }
    
    const dog = new Dog2()
    dog.cry2()
    dog.run()
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    在这里插入图片描述

    10、函数

    函数是 JavaScript 应用程序的基础,它帮助你实现抽象层,模拟类,信息隐藏和模块。

    在 TypeScript 里,虽然已经支持类,命名空间和模块,但函数仍然是主要的定义行为的地方。

    TypeScript 为 JavaScript 函数添加了额外的功能,让我们可以更容易地使用。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

    (1)基本示例(设置参数类型)

    和 JavaScript 一样,TypeScript 函数可以创建有名字的函数和匿名函数。你可以随意选择适合应用程序的方式,不论是定义一系列 API 函数还是只使用一次的函数。

    通过下面的例子可以迅速回想起这两种 JavaScript 中的函数:
    在这里插入图片描述

    function add(x: number, y: number){
        return x+y
    }
    
    let myAdd = function (x:number,y:number){
        return x + y
    }
    
    console.log(add(10,20))
    
    console.log(myAdd(20,30))
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    (2)设置返回值类型

    在这里插入图片描述

    function add(x: number, y: number):number{
        return x+y
    }
    
    let myAdd = function (x:number,y:number):number{
        return x + y
    }
    
    console.log(add(10,20))
    
    console.log(myAdd(20,30))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    在这里插入图片描述

    (3)书写完整函数类型

    现在我们已经为函数指定了类型,下面让我们写出函数的完整类型。
    在这里插入图片描述

    let myAdd2:(x:number,y:number) => number =
    function (x:number,y:number){
        return x + y
    }
    
    console.log(myAdd2(20,40))
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    在这里插入图片描述

    (4)可选参数和默认参数

    TypeScript 里的每个函数参数都是必须的。
    这不是指不能传递 null 或 undefined 作为参数,而是说编译器检查用户是否为每个参数都传入了值。
    编译器还会假设只有这些参数会被传递进函数。
    简短地说,传递给一个函数的参数个数必须与函数期望的参数个数一致。

    JavaScript 里,每个参数都是可选的,可传可不传。
    没传参的时候,它的值就是 undefined。
    在TypeScript 里我们可以在参数名旁使用 ? 实现可选参数的功能。
    比如,我们想让 lastName 是可选的:

    在 TypeScript 里,我们也可以为参数提供一个默认值当用户没有传递这个参数或传递的值是 undefined 时。
    它们叫做有默认初始化值的参数。
    让我们修改上例,把firstName 的默认值设置为 “A”。

    function buildName(firstName: string='A', lastName?: string): string {
      if (lastName) {
        return firstName + '-' + lastName
      } else {
        return firstName
      }
    }
    
    console.log(buildName('C', 'D'))
    console.log(buildName('C'))
    console.log(buildName())
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    在这里插入图片描述

    function buildName(firstName: string='A', lastName?: string): string {
      if (lastName) {
        return firstName + '-' + lastName
      } else {
        return firstName
      }
    }
    
    console.log(buildName('C', 'D'))
    console.log(buildName('C'))
    console.log(buildName())
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    在这里插入图片描述

    (5)剩余参数

    必要参数,默认参数和可选参数有个共同点:它们表示某一个参数。

    有时,你想同时操作多个参数,或者你并不知道会有多少参数传递进来。

    在 JavaScript 里,你可以使用 arguments 来访问所有传入的参数。

    在 TypeScript 里,你可以把所有参数收集到一个变量里:

    剩余参数会被当做个数不限的可选参数。 可以一个都没有,同样也可以有任意个。

    编译器创建参数数组,名字是你在省略号( …)后面给定的名字,你可以在函数体内使用这个数组。

    function info(x: string, ...args: string[]) {
      console.log(x, args)
    }
    info('abc', 'c', 'b', 'a')
    
    • 1
    • 2
    • 3
    • 4

    在这里插入图片描述
    在这里插入图片描述

    (6)函数重载

    函数重载: 函数名相同, 而形参不同的多个函数
    在JS中, 由于弱类型的特点和形参与实参可以不匹配, 是没有函数重载这一说的 但在TS中, 与其它面向对象的语言(如Java)就存在此语法

    在这里插入图片描述

    
    // 重载函数声明
    function addV (x: string, y: string): string
    function addV (x: number, y: number): number
    
    // 定义函数实现
    // @ts-ignore
    function addV(x: string | number, y: string | number): string | number {
        // 在实现上我们要注意严格判断两个参数的类型是否相等,而不能简单的写一个 x + y
        if (typeof x === 'string' && typeof y === 'string') {
            return x + y
        } else if (typeof x === 'number' && typeof y === 'number') {
            return x + y
        }
    }
    
    console.log(addV(1, 2))
    console.log(addV('a', 'b'))
    // console.log(add(1, 'a')) // error
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    在这里插入图片描述

    11、泛型

    指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定具体类型的一种特性。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

    (1)引入、使用函数泛型

    下面创建一个函数, 实现功能: 根据指定的数量 count 和数据 value ,
    创建一个包含 count 个 value 的数组 不用泛型的话,这个函数可能是下面这样:
    在这里插入图片描述

    function createArray(value: any, count: number): any[] {
      const arr: any[] = []
      for (let index = 0; index < count; index++) {
        arr.push(value)
      }
      return arr
    }
    const arr1 = createArray(11, 3)
    const arr2 = createArray('aa', 3)
    console.log(arr1[0].toFixed(), arr2[0].split(''))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    在这里插入图片描述
    使用函数泛型
    在这里插入图片描述

    function createArray2 <T> (value: T, count: number) {
        const arr: Array<T> = []
        for (let index = 0; index < count; index++) {
            arr.push(value)
        }
        return arr
    }
    const arr3 = createArray2<number>(11, 3)
    console.log(arr3[0].toFixed())
    // console.log(arr3[0].split('')) // error
    const arr4 = createArray2<string>('aa', 3)
    console.log(arr4[0].split(''))
    // console.log(arr4[0].toFixed()) // error
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    在这里插入图片描述

    (2)多个泛型参数的函数

    一个函数可以定义多个泛型参数
    在这里插入图片描述

    function swap <K, V> (a: K, b: V): [K, V] {
      return [a, b]
    }
    const result = swap<string, number>('abc', 123)
    console.log(result[0].length, result[1].toFixed())
    
    • 1
    • 2
    • 3
    • 4
    • 5

    在这里插入图片描述

    (3)泛型接口

    在定义接口时, 为接口中的属性或方法定义泛型类型
    在使用接口时, 再指定具体的泛型类型
    在这里插入图片描述

    
    interface IbaseCRUD <T> {
        data: T[]
        add: (t: T) => void
        getById: (id: number) => T
    }
    class User {
        id?: number; //id主键自增
        name: string; //姓名
        age: number; //年龄
        constructor (id?: number,name:string = '', age:number =0) {
            this.id = id
            this.name = name
            this.age = age
        }
    }
    class UserCRUD implements IbaseCRUD <User> {
        data: User[] = []
    
        add(user: User): void {
            this.data.push(user)
            console.log('保存user', user.id)
        }
    
        getById(id: number) :User {
            var user = new User();
            this.data.forEach((item:User)=>{
                if(item.id === id){
                    user = item
                }
            })
            return user
        }
    }
    const userCRUD = new UserCRUD()
    userCRUD.add(new User(1,'tom', 12))
    userCRUD.add(new User(2,'tom2', 13))
    console.log(userCRUD.data)
    console.log(userCRUD.getById(2))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39

    (4)泛型类

    在定义类时, 为类中的属性或方法定义泛型类型 在创建类的实例时, 再指定特定的泛型类型
    在这里插入图片描述

    console.log("===泛型类")
    class GenericNumber<T> {
        zeroValue?: T;
        add?: (x: T, y: T) => T
    }
    let myGenericNumber = new GenericNumber<number>()
    myGenericNumber.zeroValue = 0
    myGenericNumber.add = function(x, y) {
        return x + y
    }
    
    let myGenericString = new GenericNumber<string>()
    myGenericString.zeroValue = 'abc'
    myGenericString.add = function(x, y) {
        return x + y
    }
    
    console.log(myGenericString.add(myGenericString.zeroValue, 'test'))
    console.log(myGenericNumber.add(myGenericNumber.zeroValue, 12))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    在这里插入图片描述

    (5)泛型约束

    如果我们直接对一个泛型参数取 length 属性, 会报错, 因为这个泛型根本就不知道它有这个属性。

    // 没有泛型约束
    function fn <T>(x: T): void {
      console.log(x.length)  // error
    }
    
    • 1
    • 2
    • 3
    • 4

    在这里插入图片描述
    我们可以使用泛型约束来实现

    //没有泛型约束
    interface Lengthwise{
        length:number;
    }
    function fn2 <T extends Lengthwise>(x: T): void {
        console.log(x.length)
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    在这里插入图片描述
    在这里插入图片描述

    12、声明文件

    当使用第三方库时,我们需要引用它的声明文件,才能获得对应的代码补全、接口提示等功能

    什么是声明语句

    假如我们想使用第三方库 jQuery,一种常见的方式是在 html 中通过

    但是在 ts 中,编译器并不知道 $ 或 jQuery 是什么东西

    /* 
    当使用第三方库时,我们需要引用它的声明文件,才能获得对应的代码补全、接口提示等功能。
    声明语句: 如果需要ts对新的语法进行检查, 需要要加载了对应的类型说明代码
      declare var jQuery: (selector: string) => any;
    声明文件: 把声明语句放到一个单独的文件(jQuery.d.ts)中, ts会自动解析到项目中所有声明文件
    下载声明文件: npm install @types/jquery --save-dev
    */
    
    jQuery('#foo');
    // ERROR: Cannot find name 'jQuery'.
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    这时,我们需要使用 declare var 来定义它的类型

    declare var jQuery: (selector: string) => any;
    
    jQuery('#foo');
    
    • 1
    • 2
    • 3

    declare var 并没有真的定义一个变量,只是定义了全局变量 jQuery 的类型,仅仅会用于编译时的检查,在编译结果中会被删除。它编译结果是:

    jQuery('#foo');
    
    • 1

    一般声明文件都会单独写成一个 xxx.d.ts 文件

    创建 01_jQuery.d.ts, 将声明语句定义其中, TS编译器会扫描并加载项目中所有的TS声明文件

    declare var jQuery: (selector: string) => any;
    
    • 1

    很多的第三方库都定义了对应的声明文件库, 库文件名一般为 @types/xxx, 可以在 https://www.npmjs.com/package/package 进行搜索

    有的第三库在下载时就会自动下载对应的声明文件库(比如: webpack),有的可能需要单独下载(比如jQuery/react)

    13、内置对象

    JavaScript 中有很多内置对象,它们可以直接在 TypeScript 中当做定义好了的类型。

    内置对象是指根据标准在全局作用域(Global)上存在的对象。这里的标准是指 ECMAScript 和其他环境(比如 DOM)的标准。
    1、ECMAScript 的内置对象

    ECMAScript 的内置对象
    Boolean
    Number
    String
    Date
    RegExp
    Error
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    /* 1. ECMAScript 的内置对象 */
    let b: Boolean = new Boolean(1)
    let n: Number = new Number(true)
    let s: String = new String('abc')
    let d: Date = new Date()
    let r: RegExp = /^1/
    let e: Error = new Error('error message')
    b = true
    // let bb: boolean = new Boolean(2)  // error
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    2、BOM 和 DOM 的内置对象

    Window
    Document
    HTMLElement
    DocumentFragment
    Event
    NodeList
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    const div: HTMLElement = document.getElementById('test')
    const divs: NodeList = document.querySelectorAll('div')
    document.addEventListener('click', (event: MouseEvent) => {
      console.dir(event.target)
    })
    const fragment: DocumentFragment = document.createDocumentFragment()
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
  • 相关阅读:
    BiMPM实战文本匹配【上】
    JavaScript异步编程:提升性能与用户体验
    谐振波导光栅的严格分析
    Spring Boot 2.x源码系列【4】启动流程深入解析之启动监听器
    安卓全机型 玩机 搞机 ADB FAST 各种指令解析说明与操作
    深入讲解Netty那些事儿之从内核角度看IO模型(上)
    nacos源码深度剖析理解之注册中心
    openGL之纹理
    仙人掌之歌——大规模高速扩张(1)
    为自己量身打造一个 Rust 项目模板/脚手架
  • 原文地址:https://blog.csdn.net/qq_44757034/article/details/126616411