欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解在Vue中使用TypeScript的一些思考(實踐)

 更新時間:2018年07月06日 09:17:13   作者:三毛  
這篇文章主要介紹了詳解在Vue中使用TypeScript的一些思考(實踐),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

Vue.extend or vue-class-component

使用 TypeScript 寫 Vue 組件時,有兩種推薦形式:

  • Vue.extend():使用基礎 Vue 構造器,創(chuàng)建一個“子類”。此種寫法與 Vue 單文件組件標準形式最為接近,唯一不同僅是組件選項需要被包裹在 Vue.extend() 中。
  • vue-class-component:通常與 vue-property-decorator 一起使用,提供一系列裝飾器,能讓我們書寫類風格的 Vue 組件。

兩種形式輸出結果一致,同是創(chuàng)建一個 Vue 子類,但在書寫組件選項如 props,mixin 時,有些不同。特別是當你使用 Vue.extend() 時,為了讓 TypeScript 正確推斷類型,你將不得不做一些額外的處理。接下來,我們來聊一聊它們的細節(jié)差異。

Prop

由于組件實例的作用域是孤立的,當從父組件傳遞數據到子組件時,我們通常使用 Prop 選項。同時,為了確保 Prop 的類型安全,我們會給 Prop 添加指定類型驗證,形式如下:

export default {
 props: {
  someProp: {
   type: Object,
   required: true,
   default: () => ({ message: 'test' })
  }
 }
}

我們定義了一個 someProp,它的類型是 Object。

使用 JavaScript 時,這并沒有什么不對的地方,但當你使用 TypeScript 時,這有點不足,我們并不能得到有關于 someProp 更多有用的信息(比如它含有某些屬性),甚至在 TypeScript 看來,這將會是一個 any 類型:

這意味著我們可以使用 someProp 上的任意屬性(存在或者是不存在的)都可以通過編譯。為了防止此種情況的發(fā)生,我們將會給 Prop 添加類型注釋。

Vue.extend()

使用 Vue.extend() 方法添加類型注釋時,需要給 type 斷言:

import Vue from 'vue'

interface User {
 name: string,
 age: number
}

export default Vue.extend({
 props: {
  testProps: {
   type: Object as () => User
  }
 }
})

當組件內訪問 testProps 時,便能得到相關提示:

然而,你必須以函數返回值的形式斷言,并不能直接斷言:

export default Vue.extend({
 props: {
  testProps: {
   type: Object as User
  }
 }
})

它會給出錯誤警告,User 接口并沒有實現(xiàn)原生 Object 構造函數所執(zhí)行的方法:
Type 'ObjectConstructor' cannot be converted to type 'User'. Property 'id' is missing in type 'ObjectConstructor'.

實際上,我們可從 Prop type declaration

export type Prop<T> = { (): T } | { new (...args: any[]): T & object }

export type PropValidator<T> = PropOptions<T> | Prop<T> | Prop<T>[];

export interface PropOptions<T=any> {
 type?: Prop<T> | Prop<T>[];
 required?: boolean;
 default?: T | null | undefined | (() => object);
 validator?(value: T): boolean;
}

可知 Prop type 可以以兩種不同方式出現(xiàn):

  • 含有一個調用簽名的范型 type,該簽名返回 T;
  • 一個范型構造函數簽名,該函數創(chuàng)建指定類型 T 對象 (返回值 T & object 用于降低優(yōu)先級,當兩種方式同時滿足時取第一種,其次它還可以用于標記構造函數不應該返回原始類型)。

當我們指定 type 類型為 String/Number/Boolean/Array/Object/Date/Function/Symbol 等原生構造函數時,Prop<T> 會返回它們各自簽名的返回值。

當 type 類型為 String 構造函數時,它的調用簽名返回為 string:

// lib.es5.d.ts

interface StringConstructor {
 new(value?: any): String;
 (value?: any): string;
 readonly prototype: String;
 fromCharCode(...codes: number[]): string;
}

而這也是上文中,當指定 type 類型為 Object 構造函數時,經過 Vue 的聲明文件處理,TypeScript 推斷出為 any 類型的原因:

interface ObjectConstructor {
 new(value?: any): Object;
 (): any;
 (value: any): any;
 // 其它屬性 ....
}

類似的,當我們使用關鍵字 as 斷言 Object 為 () => User 時,它能推斷出為 User 。

從 type 第二部分可知,除傳入原生構造函數外,我們還可傳入自定義類:

此外,這里有個 PR 暴露一個更直觀的類型( Vue 2.6 版本才可以用):

props: {
 testProp: Object as PropTypes<{ test: boolean }>
}

vue-class-component

得益于 vue-propperty-decorator Prop 修飾器,當給 Prop 增加類型推斷時,這些將變得簡單:

import { Component, Vue, Prop } from 'vue-property-decorator'

@Component
export default class Test extends Vue {
 @Prop({ type: Object })
 private test: { value: string }
}

當我們在組件內訪問 test 時,便能獲取它正確的類型信息。

mixins

mixins 是一種分發(fā) Vue 組件中可復用功能的一種方式。當在 TypeScript 中使用它時,我們希望得到有關于 mixins 的類型信息。

當你使用 Vue.extends() 時,這有點困難,它并不能推斷出 mixins 里的類型:

// ExampleMixin.vue
export default Vue.extend({
 data () {
  return {
   testValue: 'test'
  }
 }
})

// other.vue
export default Vue.extend({
 mixins: [ExampleMixin],
 created () {
  this.testValue // error, testValue 不存在!
 }
})

我們需要稍作修改:

// other.vue
export default ExampleMixin.extend({
 mixins: [ExampleMixin],
 created () {
  this.testValue // 編譯通過
 }
})

但這會存在一個問題,當使用多個 mixins 且推斷出類型時,這將無法工作。而在這個 Issuse 中官方也明確表示,這無法被修改。

使用 vue-class-component 這會方便很多:

// ExampleMixin.vue
import Vue from 'vue'
import Component from 'vue-class-component'

@Component
export class ExampleMixin extends Vue {
 public testValue = 'test'
}

// other.vue
import Component, { mixins } from 'vue-class-component'
import ExampleMixin from 'ExampleMixin.vue'

@Component({
 components: {
  ExampleMixin
 }
})
export class MyComp extends mixins(ExampleMixin) {
 created () {
  console.log(this.testValue) // 編譯通過
 }
}

也支持可以傳入多個 mixins。

一些其它

做為 Vue 中最正統(tǒng)的方法(與標準形式最為接近),Vue.extends() 有著自己的優(yōu)勢,在 VScode Vetur 插件輔助下,它能正確提示子組件上的 Props:

而類做為 TypeScript 特殊的存在(它既可以作為類型,也可以作為值),當我們使用 vue-class-component 并通過 $refs 綁定為子類組件時,便能獲取子組件上暴露的類型信息:

導入 .vue 時,為什么會報錯?

當你在 Vue 中使用 TypeScript 時,所遇到的第一個問題即是在 ts 文件中找不到 .vue 文件,即使你所寫的路徑并沒有問題:

在 TypeScript 中,它僅識別 js/ts/jsx/tsx 文件,為了讓它識別 .vue 文件,我們需要顯式告訴 TypeScript,vue 文件存在,并且指定導出 VueConstructor:

declare module '*.vue' {
 import Vue from 'vue'
 export default Vue
}

但是,這引起了另一個問題,當我們導入一個并不存在的 .vue 文件時,也能通過編譯:

是的,這在情理之中。

當我嘗試在 .vue 文件中導入已存在或者不存在的 .vue 文件時,卻得到不同的結果:

文件不存在時:

文件存在時:

文件不存在時,引用 Vue 的聲明文件。文件存在時,引用正確的文件定義。

這讓人很困惑,而這些都是 Vetur 的功勞。

在這個 PR 下,我找到相關解答:這個 PR 里,Vetur 提供解析其他 .vue 文件的功能,以便能獲取正確的信息,當 .vue 文件不存在時,會讀取 .d.ts 里的信息。

參考

https://github.com/vuejs/vue/pull/5887

https://github.com/vuejs/vue/issues/7211

https://github.com/vuejs/vue/pull/6856

https://github.com/vuejs/vue/pull/5887/files/1092efe6070da2052a8df97a802c9434436eef1e#diff-23d7799dcc9e9be419d28a15348b0d99

https://github.com/Microsoft/TypeScript/blob/8e47c18636da814117071a2640ccf87c5f16fcfd/src/compiler/types.ts#L3563-L3583

https://github.com/vuejs/vetur/pull/94

相關文章

  • Vue props 單向數據流的實現(xiàn)

    Vue props 單向數據流的實現(xiàn)

    這篇文章主要介紹了Vue props 單向數據流的實現(xiàn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-11-11
  • vue 音樂App QQ音樂搜索列表最新接口跨域設置方法

    vue 音樂App QQ音樂搜索列表最新接口跨域設置方法

    這篇文章主要介紹了vue 音樂App QQ音樂搜索列表最新接口跨域設置方法,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2018-09-09
  • element-ui中導航組件menu的一個屬性:default-active說明

    element-ui中導航組件menu的一個屬性:default-active說明

    這篇文章主要介紹了element-ui中導航組件menu的一個屬性:default-active說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • 使用Nuxt.js改造已有項目的方法

    使用Nuxt.js改造已有項目的方法

    這篇文章主要介紹了使用Nuxt.js改造已有項目的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-08-08
  • 使用ElementUI中el-upload上傳文件轉base64格式

    使用ElementUI中el-upload上傳文件轉base64格式

    這篇文章主要介紹了使用ElementUI中el-upload上傳文件轉base64格式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • 基于vue.js組件實現(xiàn)分頁效果

    基于vue.js組件實現(xiàn)分頁效果

    這篇文章主要為大家詳細介紹了基于vue.js組件實現(xiàn)分頁效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-12-12
  • 解讀Vue-loader的相關知識

    解讀Vue-loader的相關知識

    這篇文章主要介紹了解讀Vue-loader的相關知識,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • vue?+?electron應用文件讀寫操作

    vue?+?electron應用文件讀寫操作

    這篇文章主要介紹了vue?+?electron應用文件讀寫操作,如果要制作的應用并不復雜,完全可以將數據存儲在本地文件當中,然后應用就可以通過這些文件進行數據的讀寫,需要的朋友參考下吧
    2022-06-06
  • 五分鐘搞懂Vuex實用知識(小結)

    五分鐘搞懂Vuex實用知識(小結)

    本篇文章主要介紹了五分鐘搞懂Vuex實用知識,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-08-08
  • Vue插槽具體用法及實例分析

    Vue插槽具體用法及實例分析

    本文主要介紹了Vue框架中插槽的使用方法和應用場景。通過具體實例分析,詳細講解了插槽的具體用法,幫助讀者深入理解Vue中插槽的使用和實現(xiàn)方式
    2023-05-05

最新評論