Bootstrap

【菜狗学前端】 初探TS(结合在vue3中的使用)

TypeScript是拥有类型的JavaScript超集,它可以编译成普通、干净、完整的JavaScript代码。我们可以将TypeScript理解成加强版的JavaScript。

简单来说:Ts是带类型语法的Js; Ts是Js的超集

TS官方网站:https://www.typescriptlang.org/

TS中文官网:https://www.tslang.cn/

一、基础语法

js已有类型:

        基本类型: number, string, boolean, null, undefined, symbol,BigInt

        引用数据类型:对象、数组、函数

ts新增类型:

        联合类型、类型别名、接口、元祖、字面量类型、枚举、void、any、泛型等

(一) 原始类型

1.简单类型 :xx

let age: number = 18;

let uname: string = "zhangsan";

let flag: boolean = false;

// null和und类型用的比较少

let nullValue: null = null;

let undefinedValue: undefined = undefined;

PS:其实最后.ts转为.js文件时void仍会变为undefined,TS中加上void一般只是为了将函数返回值和其他区分出来。

2.引用类型 :xx[]

a.数组
let arr1 = ["a", "b", "c"]

let arr3: number[] = [1, 2, 3]

let arr2: string[] = ["a", "b", "c"]

console.log(arr1, arr2, arr3);



let arr4: Array<string> = ["ok", "hello"]

console.log(arr4);

思考:数组中是可以存放多种数据类型

        联合类型:|(竖线)在TS中叫做联合类型

        即:由两个或多个其他类型组成的类型,表示可以是这些类型中的任意一种

b.函数
// 函数 给参数和返回值 指定类型

// 函数声明(具名函数)

function add(a: number, b: number): number {

    return a + b

}

console.log(add(1, 2));



// 函数表达式(匿名函数)

let add2 = (a: number, b: number): number => {

    return a + b

}

console.log(add2(1, 2));



// 箭头函数写法

type AddFn = (a: number, b: number) => number

// 通过类似箭头函数形式的语法为函数添加类型,只适用于函数表达式

let add3: AddFn = (a, b) => {

    return a + b

}

console.log(add3(3, 3));

可选参数?

使用 ? 将参数标记为可选

const fn = (n?: number) => {

}

fn()

fn(1)


函数返回值:

在js中默认返回值是und,在ts中默认返回的是void

如果指定返回值类型是 undefined 那返回值一定是 undefined

// void和undfined类型

// 如果函数没有返回值,定义函数类型时返回值的类型为void

const say = () => {

    // console.log('hello ts~');

}

console.log(say());

console.log(typeof say());

const say2 = (): void => {

    // console.log('hello ts~');

}

console.log(say2());

console.log(typeof say2());

// 区分:在js中默认返回值是und,在ts中默认返回的是void

// 如果指定返回值类型是 undefined 那返回值一定是 undefined

const say3 = (): undefined => {

    // console.log('hello ts~');

}

console.log(say3());

console.log(typeof say3());
c.对象
// 空对象

let obj1 = {}

let obj2: {} = {}



// 指定对象里面的属性和方法

// let obj4: { name: string, say(): void } = {

let obj4: { name: string; say(): void } = {

    name: "wc",    // 对象里面写方法的两种写法    // say():void    say: () => void

}

// 竖着写 可以换行

let obj5: {

    // name: string,

    name: string;

    say(): void

} = {

    name: "wc",

    say() { }

}

(二) 类型别名 type

给某个指定类型起别名

定义类型别名,遵循大驼峰命名,类似于变量

        什么时候是用类名别名?

        当同一类型(引用)被多次使用,可以通过类型别名简化

// 语法:type 类型别名=具体类型

type A = (string | number)[]

let arr9: A

arr9 = [1, "a", 3]

交叉类型 &

// type  交叉类型

type APoint2D = {

    x: number,

    y: number

}



type APoint3D = APoint2D & {

    z: number

}



let o: APoint3D = {

    x: 1,

    y: 2,

    z: 3

}

(三) 接口 interface (可继承extends)

interface 后面跟的是接口名称

注意:接口的每一行只能有一个属性或者一个方法 每一行不需要加分号

interface Person {

    name: string

    age: number

    say: () => void

}

let p: Person = {

    name: "wangcai",

    age: 18,

    say() { }

}

console.log(p);



// 接口是可继承的

// 2D坐标

interface Point2D {

    x: number

    y: number

}

let p2: Point2D = {

    x: 1,

    y: 2

}

console.log(p2);



// 如何实现继承?

// 使用extends实现接口继承,达到类型复用



// 继承后 接口A 拥有了接口B的所有属性和函数的类型声明 interface Point3D extends Point2D {    z: number}



let p3: Point3D = {

    x: 1,

    y: 2,

    z: 3

}

(四)interface和type的区别

1.type是起别名,interface 是自己发明类型 自定义类型
 

// 会报错

// type APerson = {

//     name:string

// }

// type APerson = {

//     age:number

// }



interface Person1 {

    name1: string

};

interface Person1 {

    age1: number

};

// 类型合并

const c1: Person1 = {

    name1: "wc",

    age1: 18

}

2.type不可以重复定义,interface可以重复定义会合并

3.type支持对象类型和其他类型,interface只支持对象类型

4.type复用:交叉类型 &,interface复用:可以继承 extends

二、ts在vue3中的使用

(一)ref和ts

// 手动指定类型

const count =ref<number>(0)

<template>

    <div>{{ count }}</div>

    <ul>

      <li v-for="item in list" :key="item.id">

        {{ item.name }}

      </li>

    </ul>

</template>



<script setup lang="ts">

import { ref, reactive } from "vue"



// 手动指定类型const count =ref<number>(0)



// 自动类型

const count1=ref(0)



type ToDoItem={

  id:number;

  name:string;

  done:boolean

}

// [{},{},{}]

const list=ref<ToDoItem[]>([])



setTimeout(()=>{

  list.value=[

  { id: 100, name: "吃饭", done: false },

  { id: 101, name: "睡觉", done: false },

  ]

},1000)

</script>

(二)reactive和ts

type Book={

title:string;

year?:number

}

const book:Book=reactive({

title:"ts语法基础"

})

<template>

    {{ book.title }} -- {{ book.year }}

</template>



<script setup lang="ts">

import { ref, reactive } from "vue"



// 1.手动指定类型

type Book={

  title:string;

  year?:number

}const book:Book=reactive({

  title:"ts语法基础"

})

book.year=2023



// 2.自动类型推导

const Book=reactive({title:"ts语法基础2"})

</script>

(三)computed和ts

const doubleCountB = computed<string(() => (count.value * 2).toFixed(2))

<template>

  {{ count }} -- {{ doubleCount }} -- {{ doubleCountB }}

  {{ typeof count }} -- {{ typeof doubleCount }} -- {{ typeof doubleCountB }}

</template>



<script setup lang="ts">

import { ref, computed } from "vue"

const count = ref(205)



// 1.自动类型推导

// toFixed(2)保留2位小数,且返回值类型为string

const doubleCount = computed(() => (count.value * 2).toFixed(2))

// 2.指定计算属性的类型

const doubleCountB = computed<string>(() => (count.value * 2).toFixed(2))

</script>

(四)defineprops和ts(父传子)

defineProps<{

money:number;

car?:string

}>()

Child.vue

<template>

    <div class="">

      <p>{{ money }}</p>

      <p>{{ car }}</p>

    </div>

</template>



<script setup lang="ts">

// vue3基本写法

// defineProps({

//     money:{

//         type:Number,

//         required:true

//     }

// })



// ts基本语法

// defineProps<{

//     money:number;

//     car?:string

// }>()



// ts+有默认值的写法

withDefaults(

    defineProps<{

        money:number;

        car?:string

    }>(),{

        money:666,

        car:"xiaomi"

    }

)

</script>

App.vue

<template>

 <div>

  <!-- 父传子 -->

  <Child :money="money"></Child>

 </div>

</template>



<script setup lang="ts">

import { ref} from "vue"

import Child from "./components/Child.vue"

const money=ref(666)

const car=ref('大众')

</script>

(五)defineEmits和ts(子传父)

const emit =defineEmits<{

(e:"changeMoney",money:number):void;

(e:"changeCar",car:string):void;

}>()

Child.vue

<template>

    <p>{{ money }}</p>

    <button @click="emit('changeMoney', 800)">改money</button>

    <p>{{ car }}</p>

    <button @click="emit('changeCar', '宝马')">改car</button>

</template>



<script setup lang="ts">

// vue3基本写法

// const emit = defineEmits(["changeMoney", "changeCar"])



// ts写法

defineProps<{

  money: number;

  car?: string;

}>()

const emit =defineEmits<{

    (e:"changeMoney",money:number):void;

    (e:"changeCar",car:string):void;

}>()

</script>

App.vue

<template>

 <!-- 子传父 -->

 <Child

 :money="money"

 :car="car"

 @change-money="money=$event"

 @change-car="car=$event"

 ></Child>

</template>



<script setup lang="ts">

import { ref} from "vue"

import Child from "./components/Child.vue"

const money=ref(666)

const car=ref('大众')

</script>

(六)事件处理和ts :Event

as 断言​​​​​​​

e:Event

<template>

    <input type="text" @change="handleChange" value="hi">

</template>



<script setup lang="ts">

// 不处理类型

// const handleChange=(e)=>{

//     console.log(e.target.value)

// }



const handleChange=(e:Event)=>{

    // as 断言

    // e.target 事件源

    console.log((e.target as HTMLInputElement).value);

}

</script>

(七)ref写在标签上获取 :HTMLInputElement

const input = ref<HTMLInputElement | null(null)

<template>

    <div>

        <input type="text" ref="input">

    </div>

</template>



<script setup lang="ts">

import { onMounted, ref } from 'vue'

const input = ref<HTMLInputElement | null>(null)



// 页面加载完毕调用钩子函数

onMounted(()=>{

    input.value?.focus()

})

</script>

(八)非空断言 !

// ! 非空断言(方法一)

input.value!.value = "456" //可以赋值

// 类型守卫(方法二)

if (input.value) {

input.value.value = "789" //可以赋值

}

<template>

    <div>

        <input type="text" ref="input" value="123">

    </div>

</template>



<script setup lang="ts">

import { onMounted, ref } from 'vue'

const input = ref<HTMLInputElement | null>(null)



// 页面加载完毕调用钩子函数

onMounted(() => {

    // 自动获取焦点

    input.value?.focus()

    // 使用ES6 输入框没有获取到,对于可能出现null,

    console.log(input.value?.value);

    // ?可选参数 可选链  报错 因为不能赋值

    // input.value?.value = "456"



    // ! 非空断言(方法一)

    input.value!.value = "456" //可以赋值

    // 类型守卫(方法二)

    if (input.value) {

        input.value.value = "789" //可以赋值

    }

})

</script>
;