最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • typescript 学习笔记

    正文概述 掘金(瑟曦爱唐僧)   2021-08-04   694

    typescript 学习笔记

    记录和学习下ts的基本使用和以供后续排查问题和作为参考手册,单纯看文档比较枯燥,记录一下简单的上手使用的点。

    安装

    npm install -g typescript
    yarn global add typescrpt
    
    mkdir tsproject && cd tsproject && touch index.ts >> echo "console.log(1)"
    
    // 手动编译这个文件
    tsc index.ts
    
    // 初始化ts项目配置文件
    tsc --init
    
    // 修改 tsconfig.json
     "outDir": "./src", 
    
    // 或vscode 终端 > 运行任务 > ts > tsconfig.json
    tsc -w 
    

    ts 的核心功能就是四个字:类型约束

    核心功能

    数据类型

    原始数据类型包括:布尔值、数值、字符串、null、undefined 以及 ES6 中的新类型 SymbolES10 中的新类型 BigInt

    // 布尔
    var flag: Boolean = true;
    
    // number
    var num1: Number = 100;
    var num2: Number = 12.3;
    
    // string
    var str1: String = "anikin";
    
    let myFavoriteNumber = 'seven';
    myFavoriteNumber = 7;  // 会报错,类型推论,等价于let myFavoriteNumber: string = 'seven';
    
    // array
    var arr1: number[] = [1, 2, 3, 4];
    var arr2: Array<number> = [1, 2, 3, 4]; // 泛型
    var arr3: [string, number] = ["anikin", 100]; // 元组类型
    
    // enum 枚举类型:将变量的值一一列举出来,变量的取值只限于这些范围
    enum Flag {
      default, // 默认值是index或者是上一个值+1  0
      success = 1,
      error = -1,
      last, // 0
    }
    var f1: Flag = Flag.success;
    // console.log(Flag.default, Flag.last);
    
    enum Days {Sun, Mon, Tue, Wed, Thu, Fri, Sat};
    // Days['Sun'] = 0;
    
    // any 用处
    var box:any = document.getElementById("app");
    box.style.color = "red";
    
    // undefined || null
    var usrinfo: undefined;
    var num3: number | undefined;
    num3 = 100;
    

    函数

    // 函数声明(Function Declaration)
    function show(): void {} // 无返回值
    function conso(name: String, age: number): string {
      return `${name}===${age}`;
    }
    
    // 函数表达式(Function Expression)
    const myAdd2 = (x: number, y: number) => x + y;
    
    // 匿名
    var getInfo = function (): void {};
    
    // 可选参数 age 是可选参数
    // 可选必须放到最后面
    // name 是默认参数, 默认参数也是可选参数,如果不是最后一位的还需传参
    function conso2(name: String = "zhangsan", age?: number): string {
      return `${name}===${age}`;
    }
    
    
    // 实现函数求和  ...rest:number[] 展开的数组
    // 剩余参数 init = 1
    function add(initValue: number, ...rest: number[]): number {
      var res = initValue;
      for (let index = 0; index < rest.length; index++) {
        res += rest[index];
      }
      return res;
    }
    //  add(1,2,3,4,5) 
    
    // 重载允许一个函数接受不同数量或类型的参数时,作出不同的处理。
    function reverse(x: number): number;
    function reverse(x: string): string;
    function reverse(x: number | string): number | string | void {
        if (typeof x === 'number') {
            return Number(x.toString().split('').reverse().join(''));
        } else if (typeof x === 'string') {
            return x.split('').reverse().join('');
        }
    }
    

    修饰符:

    • public 修饰的属性或方法是公有的,可以在任何地方被访问到,属性和方法默认。
    • private 修饰的属性或方法是私有的,不能在声明它的类的外部访问。
    • protected 修饰的属性或方法是受保护的,它和 private 类似,区别是它在子类中也是允许被访问的
    class Person {
      // 默认是public
      public name: string;
      static ajaxName: string = "ajaxNameSpace"; // 静态属性
    
      constructor(n: string) {
        this.name = n;
      }
    
      // 静态方法可以访问静态属性
      static ajax() {
        return Person.ajaxName;
      }
    
      run(): void {
        console.log(this.name);
      }
    
      // 多态: 父级定义接口但是不去实现,而是由集成它的类去实现这个功能
      // 每一个子类都有不同的表现
      size(): number {
        return 100;
      }
    }
    
    class Child extends Person {
      private age: number = 100;
      readonly school = '北大附小'; // 只读属性
      // super 相当于调用父的构造函数
      // 初始化父类
      constructor(n: string) {
        super(n);
      }
    
      protected work(): string {
        return "work";
      }
    
      run(): void {
        console.log("child run" + this.work());
      }
    }
    
    var c1 = new Child("zhangsan");
    
    // ts 层面无法访问,但是最终都是转义成为es5的话都是可以运行的,记住:ts只是类型约束
    // console.log(c1.work());
    // console.log(c1.age);
    
    
    // 抽象类一般是用来定义标准的
    abstract class National {
      // 抽象方法只能出现在抽象类里面,抽象类只能被继承实现,无法直接实例化
      abstract language(): any;
    }
    
    class China extends National {
      public lan: string = "汉语";
      language() {
        return this.lan;
      }
    }
    
    var ch = new China();
    
    

    接口

    接口:行为和动作的规范,对批量的方法进行约束。

    和抽象类的区别:

    • 抽象类里面可以有方法的实现,但是接口完全都是抽象的,不存在方法的实现;
    • 子类只能继承一个抽象类,而接口可以被多个实现;
    • 抽象方法可以是publicprotected,但是接口只能是public,默认的;
    • 抽象类可以有构造器,而接口不能有构造器
    // 属性接口 约束json
    interface FullName {
      firstName: string;
      secondName: string;
      age?: number;  // 可选属性
    }
    
    interface ajaxConfig {
      url: string;
      dataType: string;
      type: string;
      data?: string;
    }
    
    // 约束函数类型  返回的类型也是 string
    interface encrypt {
      (key: string, value: number): string;
    }
    
    // 类的类型接口  跟抽象类很像
    interface Animal {
      name: string;
      eat(n: string): void;
    }
    
    // 接口的继承实现
    interface Cattype extends Animal {
      color: string;
      say(): string;
    }
    
    // 
    interface Light {
        lightOn(): void;
        lightOff(): void;
    }
    
    
    // 接口扩展:接口可以继承接口,很好理解,耦合多个接口到实现一个聚合接口
    interface Alarm {
        alert(): void;
    }
    
    interface LightableAlarm extends Alarm {
        lightOn(): void;
        lightOff(): void;
    }
    
    // 不使用接口
    function printLabel(labelInfo: { labels: string }): void {
      console.log(labelInfo);
    }
    
    // 使用接口
    function printLabel2(name: FullName): void {
      console.log("=====printLabel2", name);
    }
    
    
    function ajax(config: ajaxConfig) {
      var xhr = new XMLHttpRequest();
      xhr.open(config.type, config.url);
      xhr.send(config.data);
      xhr.onload = function (res) {
        console.log(res);
      };
    }
    
    
    // 函数类型的接口
    // 基于encrypt这个接口,实现加密的函数类型接口
    var md5: encrypt = function (key: string, value: number): string {
      return window.btoa(key) + value;
    };
    
    
    // 类实现2个接口
    class Dog implements Animal,Light {
      public name: string = "little kitty";
      eat(food: string): void {
        console.log(this.name + "eat=", food);
      }
       lightOn() {
          console.log('Car light on');
       }
       lightOff() {
          console.log('Car light off');
       }
    }
    
    

    泛型

    泛型(Generics) 是指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定类型的一种特性。

    解决类 接口 方法的复用性 以及对不特定数据类型的支持。

    // 1: 泛型函数
    
    // 同时返回string 和 number 类型 支持多种类型的函数,传入什么类型 返回什么类型
    // <T> 表示泛型
    // 泛型和any 相比:有类型校验
    
    function getData<T>(value: T): T {
      return value;
    }
    // getData<number>(100);
    
    
    function getLength<T>(arr: T[]): T[] {
      console.log(arr.length); // must be exist;
      return arr;
    }
    // getLength<number>([1, 2]);
    
    // 1: 泛型类
    
    // 实例化的时候传入的这个类型来决定的
    class MinClass<T> {
      list: T[] = [];
      add(value: T): void {
        this.list.push(value);
      }
      min(): T {
        return this.list.sort()[0];
      }
    }
    
    // var m1 = new MinClass<number>();
    // m1.add(10);
    // m1.add(20);
    // m1.add(2);
    // m1.add(21);
    // m1.add(1);
    // m1.add(9);
    // console.log(m1.min());
    
    // 3: 泛型接口
    interface ConfigData {
      // (value: string, key: number): string;
      <T>(value: T): T;
    }
    var gd: ConfigData = function <T>(value: T): T {
      return value;
    };
    
    interface GenericIdentityFn<T> {
      (arg: T): T;
    }
    var myIdentity: GenericIdentityFn<number> = function <T>(arg: T): T {
      return arg;
    };
    
    /**
     * 实现案例: 把类当做参数约束类型,这样我们就能清楚的知道使用的具体是哪个泛型类型
     */
    
    // 定义一个用户类:实现和数据库字段做一个映射
    class User {
      username: string | undefined;
      passwd: string | undefined;
      constructor(u: string, p: string) {
        this.username = u;
        this.passwd = p;
      }
    }
    
    // 文章类
    class Arcticle {
      title: string | undefined;
      desc: string | undefined;
      status?: number = 0;
      constructor(t: string, d: string, s: number = 0) {
        this.title = t;
        this.desc = d;
        this.status = s;
      }
    }
    
    // 数据库实现封装类
    class MysqlDb<T> {
      // 把类当做参数传递
      add(data: T): boolean {
        console.log(data);
        return true;
      }
    }
    
    var u = new User("zhangsan", "231231");
    var arc = new Arcticle("吃饭", "吃饭的描述");
    
    // var db = new MysqlDb<User>();
    var db = new MysqlDb<Arcticle>();
    
    db.add(arc);
    

    案例:实现一个mysql和mondb等业务层的封装

    /**
     * 封装一个mysql 和 mongdb 统一的封装
     */
    const store: any = {};
    
    // curd 接口
    interface DBI<T> {
      add(data: T): boolean;
      update(d1: T, d2: number): boolean;
      delete(d: number): boolean;
      get(id: number): any[];
    }
    
    // 要实现泛型接口 这类也必须是泛型
    class Mysql<T> implements DBI<T> {
      add(data: any): boolean {
        console.log(data);
        let len = Object.keys(store);
        let id = len.length + 1;
        store[id] = data;
        return true;
      }
      update(d1: any, d2: number): boolean {
       return true;
      }
      delete(d: number): boolean {
        return true;
      }
      get(id: number): any[] {
        return store[id];
      }
    }
    
    // 定义一个用户类和数据库做映射
    class Users {
      username: string | undefined;
      passwd: number | undefined;
      constructor(u: string, p: number) {
        this.username = u;
        this.passwd = p;
      }
    }
    
    var p1 = new Users("zhangsan", 123123123);
    var p2 = new Users("zhaosi", 11111);
    
    var db2 = new Mysql<Users>();
    db2.add(p1);
    
    

    命名空间

    命名空间一个最明确的目的就是解决重名问题。

    /// <reference path = "SomeFileName.ts" /> 的方式3.x之后推荐es模块化导出和引入。

    com.ts:

    //  命名空间 当成一个模块
    export namespace A {
      interface Animal {
        name: string;
        eat(): void;
      }
    
      export class Dog2 implements Animal {
        name: string;
        constructor(n: string) {
          this.name = n;
        }
    
        eat() {
          console.log("==========dog eat=====");
        }
      }
    
      export class Cat implements Animal {
        name: string;
        constructor(n: string) {
          this.name = n;
        }
    
        eat() {
          console.log("==========cat eat=====");
        }
      }
    }
    

    index.ts:

    import { A } from "./com";
    var cc = new A.Cat("bb");
    

    装饰器

    随着 TypeScriptES6里引入了类,在一些场景下我们需要额外的特性来支持标注或修改类及其成员。 装饰器(Decorators)为我们在类的声明及成员上通过元编程语法添加标注提供了一种方式。

    若要启用实验性的装饰器特性,你必须在命令行或tsconfig.json里启用experimentalDecorators编译器选项。

    装饰器是一种特殊类型的声明,它能够被附加到类声明,方法, 访问符,属性或参数上。,分为 普通装饰器装饰器工厂(可传参)

    // ts 中装饰器就是一个方法,可以注入类 方法 属性参数
    
    // 普通装饰 类装饰器 param:当前的类 Animal
    function logClass(param: any) {
      param.prototype.log = function () {
        console.log("log====");
      };
    }
    @logClass
    class Animal{
    }
    
    
    // 装饰器工厂
    function appConfig(param: string) {
      return function (target: any) {
        // console.log(target); // HttpClient
        // console.log(param);  // 手动传的参: http://www.baidu.com/
      };
    }
    
    @appConfig("http://www.baidu.com/")
    class App {
    }
    
    
    // 属性装饰器,装饰属性使用。 有两个参数
    function changeUrl(parm: string) {
      // target 被装饰的类  atrr 被装饰的属性
      return function (target: any, attr: any) {
        console.log(target, attr);
        target[attr] = parm;
      };
    }
    
    class App2{
       @changeUrl("http://git.100tal.com/")
       public url: any | undefined;
    }
    
    
    // 方法装饰器: 装饰方法
    function get(param: any) {
     // fname 函数名称 getData
     // target:对象 HttpClient
     // desc: configurable: true,enumerable: true,value: ƒ (),writable: true
    
    return function (target: any, fname: string, desc: any) {    
        // 重写了被装饰的函数,被装饰的函数就不是再执行了
        var oMthod = desc.value; // === getData
        desc.value = function (...args: any) {
          // args 捕获外面被装饰函数的参数 [100, 200, 300]
          args = args.map((i: any) => String(i));
          console.log("xxx", args);
    
          // 不要重写 而是扩展了之前的方法,这样子两个都会执行
          oMthod.apply(this, args);
        };
      };
    }
    
    class HttpClient {
      public url: any | undefined;
      constructor() {}
    
      @get(100)
      getData() {
        console.log("getData===", this.url);
      }
    }
    
    var http: any = new HttpClient();
    http.getData(100, 200, 300);
    

    起源地下载网 » typescript 学习笔记

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元