精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Vue 3.0 進階之依賴注入探秘

開發 前端
為了讓大家能夠更深入地理解 provide/inject 特性,阿寶哥從源碼角度分析了 provide 和 inject 函數的具體實現。

 [[384053]]

本文轉載自微信公眾號「全棧修仙之路」,作者阿寶哥。轉載本文請聯系全棧修仙之路公眾號。  

本文是 Vue 3.0 進階系列 的第六篇文章,在這篇文章中,阿寶哥將帶大家一起探索 Vue 3 中的依賴注入功能。

使用過 Angular 的小伙伴對 依賴注入 應該不會陌生,依賴注入簡稱為 DI(Dependency Injection)。組件之間的依賴關系由容器在運行期決定,形象的說,即由容器動態的將某個依賴關系注入到組件之中。依賴注入的目的并非為軟件系統帶來更多功能,而是為了提升組件重用的頻率,并為系統搭建一個靈活、可擴展的平臺。

在 Vue 3.0 中,為我們提供了簡單的依賴注入功能 —— provide/inject。它們解決了以下問題:有一些深度嵌套的組件,而深層的子組件只需要父組件的部分內容。在這種情況下,如果仍然將 prop 沿著組件鏈逐級傳遞下去,這樣使用起來會很麻煩。

為了解決上述問題,Vue 提供了 provide 和 inject。使用 provide/inject 之后,無論組件層次結構有多深,父組件都可以作為其所有子組件的依賴提供者。

(圖片來源 —— https://v3.cn.vuejs.org/guide/component-provide-inject.html)

由上圖可知,在父組件上通過 provide 提供數據,子組件通過 inject 注入數據。介紹完 provide/inject 的作用之后,我們來看一下具體的示例。

一、Provide/Inject 使用示例

在使用 provide/inject 特性時,你可以通過 provide 和 inject 選項的方式來使用它。這種方式官方文檔已經有介紹了,這里阿寶哥將介紹另一種使用方式,即在組合式 API 的 setup 組件選項中,通過 provide 和 inject 函數的方式來實現依賴注入。

  1. <div id="app"></div> 
  2. <script> 
  3.    const { createApp, h, provide, inject } = Vue 
  4.    const app = createApp({ 
  5.      render: () => h(Provider) 
  6.    }) 
  7.     
  8.    const Provider = { 
  9.      setup() { 
  10.        provide('name''阿寶哥'
  11.        return () => h(Middle) 
  12.      } 
  13.    } 
  14.  
  15.    const Middle = { 
  16.      render: () => h(Consumer) 
  17.    } 
  18.  
  19.    const Consumer = { 
  20.      setup() { 
  21.        const name = inject('name'
  22.        return () => `大家好,我是${name}!` 
  23.      } 
  24.    } 
  25.    app.mount('#app')  
  26. </script> 

在以上示例中,在 Provider 組件內通過 provide 函數配置了數據,而在 Consumer 組件中通過 inject 函數獲取 Provider 組件中已配置的數據。需要注意的是,示例中的 Consumer 組件是作為 Provider 組件的孫組件。因此,通過使用 provide/inject 提供的依賴注入功能,我們實現了數據的跨層級傳遞。

介紹完 provide 和 inject 函數的基本使用之后,接下來阿寶哥將帶大家一起來揭開它們背后的秘密。

二、Provide 函數

在分析 provide 函數之前,我們先來回顧一下它的用法:

  1. const Provider = { 
  2.   setup() { 
  3.     provide('name''阿寶哥'
  4.     return () => h(Middle) 
  5.   } 

該函數被定義在 runtime-core/src/apiInject.ts 文件中:

  1. // packages/runtime-core/src/apiInject.ts 
  2. export interface InjectionKey<T> extends Symbol {} 
  3.  
  4. export function provide<T>(key: InjectionKey<T> | string | number, value: T) { 
  5.   if (!currentInstance) { 
  6.     if (__DEV__) { 
  7.       warn(`provide() can only be used inside setup().`) 
  8.     } 
  9.   } else { 
  10.     let provides = currentInstance.provides 
  11.     // 默認情況下,組件實例會繼承于它父組件實例的 provides 對象,當它需要為本身提供 
  12.     // 值是,它將使用父組件的 provides 對象作為原型對象來創建屬于它自己的 provides 
  13.     // 對象。這樣的話,`inject` 函數就可以簡單地從直接父對象中查找需注入的值,并讓原型鏈 
  14.     // 來完成這個工作。 
  15.     const parentProvides = 
  16.       currentInstance.parent && currentInstance.parent.provides 
  17.     if (parentProvides === provides) { 
  18.       provides = currentInstance.provides = Object.create(parentProvides) 
  19.     } 
  20.     // TS 不允許使用 symbol 作為索引類型 
  21.     provides[key as string] = value 
  22.   } 

通過觀察以上的代碼,我們可以得出以下結論:

  • provide 函數只能使用在組合式 API 的 setup 函數中。
  • 組件實例上會有一個 provides 屬性,通過 provide 配置的數據,最終會被保存到組件的 provides 屬性中。
  • provide 函數支持 3 種類型作為 key,即 InjectionKey | string | number,其中 InjectionKey 類型是 Symbol 類型的子類型。

在以上代碼中,我們見到了 currentInstance 對象,那么這個對象內部的結構是怎樣的呢?為了一探究竟,我們在 provide 函數內部加個斷點:

由上圖可知, currentInstance 是一個含有多種屬性的普通對象。其中 bc(BEFORE_CREATE)、bm(BEFORE_MOUNT) 和 bu(BEFORE_UPDATE) 是與生命周期相關的鉤子。那么問題又來了,currentInstance 對象是怎么創建的?看過之前 Vue 3.0 進階系列文章的小伙伴,可能對 createComponentInstance 函數有點印象,該函數的作用就是創建組件實例,具體代碼如下所示:

  1. // packages/runtime-core/src/component.ts 
  2. export function createComponentInstance( 
  3.   vnode: VNode, 
  4.   parent: ComponentInternalInstance | null
  5.   suspense: SuspenseBoundary | null 
  6. ) { 
  7.   const type = vnode.type as ConcreteComponent 
  8.   // inherit parent app context - or - if root, adopt from root vnode 
  9.   const appContext = 
  10.     (parent ? parent.appContext : vnode.appContext) || emptyAppContext 
  11.  
  12.   const instance: ComponentInternalInstance = { 
  13.     uid: uid++, vnode, type, 
  14.     parent, appContext, 
  15.     root: null!, // to be immediately set 
  16.     subTree: null!, // will be set synchronously right after creation 
  17.     updatenull!, // will be set synchronously right after creation 
  18.     render: null, effects: null
  19.     provides: parent ? parent.provides : Object.create(appContext.provides), 
  20.     bc, bm, bu 
  21.     // 省略大部分屬性 
  22.   } 
  23.   // 省略部分代碼 
  24.   instance.root = parent ? parent.root : instance 
  25.   instance.emit = emit.bind(null, instance) 
  26.   return instance 

需要注意的是,如果當前組件 parent 屬性的值不為 null 時,則將使用 parent.provides 的值作為組件實例 provides 屬性的屬性值。介紹完 provide 和 createComponentInstance 函數,為了讓大家能夠更好地理解前面的示例,阿寶哥用一張圖來總結一下示例中組件之間的關系:

對于根組件來說,它的 parent 屬性值為 null。好的,provide 函數就先介紹到這里,下面我們來開始介紹 inject 函數。

三、Inject 函數

同樣,在分析 inject 函數之前,我們也先來回顧一下它的用法:

  1. const Consumer = { 
  2.   setup() { 
  3.     const name = inject('name'
  4.     return () => `大家好,我是${name}!` 
  5.   } 

inject 函數與 provide 函數是互相配合的,它們都被定義在 runtime-core/src/apiInject.ts 文件中:

  1. // packages/runtime-core/src/apiInject.ts 
  2. export function inject( 
  3.   key: InjectionKey<any> | string, 
  4.   defaultValue?: unknown, 
  5.   treatDefaultAsFactory = false 
  6. ) { 
  7.   const instance = currentInstance || currentRenderingInstance // 獲取當前實例 
  8.   if (instance) { 
  9.     // to support `app.use` plugins, 
  10.     // fallback to appContext's `provides` if the intance is at root 
  11.     const provides = 
  12.       instance.parent == null 
  13.         ? instance.vnode.appContext && instance.vnode.appContext.provides 
  14.         : instance.parent.provides 
  15.  
  16.     if (provides && (key as string | symbol) in provides) { 
  17.       // TS doesn't allow symbol as index type 
  18.       return provides[key as string] 
  19.     } else if (arguments.length > 1) { 
  20.       return treatDefaultAsFactory && isFunction(defaultValue) 
  21.         ? defaultValue() 
  22.         : defaultValue 
  23.     } else if (__DEV__) { 
  24.       warn(`injection "${String(key)}" not found.`) 
  25.     } 
  26.   } else if (__DEV__) { 
  27.     warn(`inject() can only be used inside setup() or functional components.`) 
  28.   } 

在 inject 函數中,我們可以清楚地看到如果當前實例的 parent 屬性為 null 時,則會從 appContext 上下文中獲取 provides 對象,否則將從當前實例的父組件實例中獲取 provides 對象。

對于我們的示例來說,在獲取到 provides 對象后,就會判斷 name 屬性是否存在于當前的 provides 對象中,此時該對象是 { name: "阿寶哥"},所以會直接返回 "阿寶哥"。

此外,通過觀察 inject 函數,我們還可以得出以下結論:

  • inject 函數的第二個參數是一個可選參數 —— defaultValue?: unknown,用于設置默認值或默認值工廠。即在 provides 對象上找不到 key 對應值的時候,可以使用默認值或默認值工廠的返回值來代替。
  • inject 函數只能在 setup() 或函數組件中使用。

四、App 對象中的 provide API

在創建完 Vue 3 應用對象之后,我們可以使用該對象提供的 provide 方法。該方法設置一個可以被注入到應用范圍內所有組件中的值。之后,組件就可以使用 inject 來接收 provide 的值。

  1. import { createApp } from 'vue' 
  2.  
  3. const app = createApp({ 
  4.   inject: ['name'], 
  5.   template: ` 
  6.     <div> 
  7.       {{ name }} 
  8.     </div> 
  9.   ` 
  10. }) 
  11.  
  12. app.provide('name''阿寶哥'

需要注意的是,app.provide 方法不應該與 provide 組件選項或組合式 API 中的 provide 方法混淆。雖然它們也是相同的 provide/inject 機制的一部分,但是是用來配置組件 provide 的值而不是應用 provide 的值。

介紹完 app.provide 方法之后,我們來了解一下它的實現。看過 ”Vue 3.0 進階“ 系列教程的小伙伴,對 app 對象應該不會陌生。因為在前面的文章中,阿寶哥已經介紹過 component、directive 和 mount 等方法。接下來,我們來看一下 provide 方法的具體實現:

  1. // packages/runtime-core/src/apiCreateApp.ts 
  2. export function createAppAPI<HostElement>( 
  3.   render: RootRenderFunction, 
  4.   hydrate?: RootHydrateFunction 
  5. ): CreateAppFunction<HostElement> { 
  6.   return function createApp(rootComponent, rootProps = null) { 
  7.     const context = createAppContext() 
  8.  
  9.     // 省略大部分內容 
  10.     const app: App = (context.app = { 
  11.       _uid: uid++, 
  12.       _context: context, 
  13.  
  14.       provide(key, value) { 
  15.         // TypeScript doesn't allow symbols as index type 
  16.         // https://github.com/Microsoft/TypeScript/issues/24587 
  17.         context.provides[key as string] = value 
  18.         return app 
  19.       } 
  20.     }) 
  21.  
  22.     return app 
  23.   } 

由以上代碼可知,在 provide 方法內部會把 key 和 value 以鍵值對的形式保存在應用上下文 context 對象的 provides 屬性中。

  1. // packages/runtime-core/src/apiCreateApp.ts 
  2. export function createAppContext(): AppContext { 
  3.   return { 
  4.     app: null as any
  5.     config: { ... }, 
  6.     mixins: [], 
  7.     components: {}, 
  8.     directives: {}, 
  9.     provides: Object.create(null
  10.   } 

五、阿寶哥有話說

5.1 在嵌套的 providers 場景下,存在同名的 key 會怎么樣?

  1. <div id="app"></div> 
  2. <script> 
  3.    const { createApp, h, provide, inject } = Vue 
  4.    const app = createApp({ 
  5.      render: () => h(ProviderOne) 
  6.    }) 
  7.     
  8.    const ProviderOne = { 
  9.      setup() { 
  10.        provide('foo''foo'
  11.        provide('bar''bar'
  12.        return () => h(ProviderTwo) 
  13.      } 
  14.    } 
  15.  
  16.    const ProviderTwo = { 
  17.      setup() { 
  18.        provide('foo''fooOverride'
  19.        provide('baz''baz'
  20.        return () => h(Consumer) 
  21.       } 
  22.    } 
  23.  
  24.    const Consumer = { 
  25.      setup() { 
  26.        const foo = inject('foo'
  27.        const bar = inject('bar'
  28.        const baz = inject('baz'
  29.        return () => [foo, bar, baz].join(','
  30.      } 
  31.    } 
  32.    app.mount('#app')  
  33. </script> 

在以上代碼中,ProviderOne 和 ProviderTwo 組件中使用同樣的 foo 屬性名配置了 Provider。然后,我們在底層的 Consumer 組件中使用 inject API 分別注入了 ProviderOne 和 ProviderTwo 中配置的值。接下來,我們先來看一下結果:

  1. fooOverride,bar,baz 

由以上結果可知,在嵌套 providers 的場景中,會就近從父組件實例中獲取對應的值,找不到的話,會往上一層層進行查找。

5.2 通過 inject 獲取響應式的值,能否正常工作?

在某些場景下,我們希望往深層的子組件傳遞通過響應式 API 創建的響應式的值,那么通過 inject 函數獲取的響應式的值可以正常工作么?要弄清楚這個問題,我們來看一個具體的示例:

  1. <div id="app"></div> 
  2. <script> 
  3.    const { createApp, h, provide, inject, ref, onMounted } = Vue 
  4.    const nameRef = ref("阿寶哥"); 
  5.    const app = createApp({ 
  6.      setup() { 
  7.        provide("name", nameRef); 
  8.        return () => h(Middle) 
  9.      } 
  10.     }) 
  11.  
  12.    const Middle = { 
  13.      render: () => h(Consumer) 
  14.    } 
  15.  
  16.    const Consumer = { 
  17.      setup() { 
  18.        const name = inject('name'
  19.        onMounted(() => { 
  20.          setTimeout(() => nameRef.value = "kakuqo", 2000); 
  21.        }) 
  22.        return () => `大家好,我是${name.value}!` 
  23.      } 
  24.   } 
  25.   app.mount('#app')  
  26. </script> 

在以上代碼中,我們通過 ref API 創建了一個 nameRef 對象,然后在根組件中通過 provide 函數配置相應的 Provider。而在 Consumer 組件的 setup 方法內,我們通過 inject 函數注入了 nameRef 對象,并通過 name.value 訪問了該對象內保存的值。

此外,在 setup 方法內部,我們還使用了 onMounted 生命周期鉤子,在鉤子對應的回調函數中,我們延遲 2S 修改 nameRef 對象的值。

以上示例成功運行后,首先會先顯示 大家好,我是阿寶哥!,差不多 2S 后頁面會刷新為 大家好,我是kakuqo!。

5.3 是否支持 self-inject?

什么是 self-inject 呢?這里阿寶哥不做過多解釋,我們直接來看個具體的例子:

  1. <div id="app"></div> 
  2. <script> 
  3.    const { createApp, h, provide, inject } = Vue 
  4.    const app = createApp({ 
  5.      render: () => h(Provider) 
  6.    }) 
  7.     
  8.    const Provider = { 
  9.      setup() { 
  10.        provide('name''阿寶哥'
  11.        const injectedName = inject('name'
  12.        return () => h(injectedName) 
  13.      } 
  14.    } 
  15.    app.mount('#app')  
  16. </script> 

在以上代碼中,我們在 Provider 組件的 setup 方法內部先使用 provide 函數配置了相應的 Provider,然后使用 inject 函數來獲取對應的值。很明顯這個操作并沒有實際的意義,那么可以這樣使用么?答案是可以的,以上示例成功運行之后,Provider 組件會被轉換為 注釋節點。

  1. <div id="app" data-v-app=""><!----></div> 

那么為什么會轉為注釋節點呢?因為 injectedName 的值為 undefined,在通過 h 函數創建 VNode 對象的時候,會繼續調用 createVNode 函數,在該函數內部如果發現是 type 類型為 falsy 值時,會把 VNode 對象的類型統一轉換為 Comment 類型。

  1. // packages/runtime-core/src/vnode.ts 
  2. function _createVNode( 
  3.   type: VNodeTypes | ClassComponent | typeof NULL_DYNAMIC_COMPONENT, 
  4.   props: (Data & VNodeProps) | null = null
  5.   children: unknown = null
  6.   patchFlag: number = 0, 
  7.   dynamicProps: string[] | null = null
  8.   isBlockNode = false 
  9. ): VNode { 
  10.   if (!type || type === NULL_DYNAMIC_COMPONENT) { 
  11.     if (__DEV__ && !type) { 
  12.       warn(`Invalid vnode type when creating vnode: ${type}.`) 
  13.     } 
  14.     type = Comment 
  15.   } 
  16.   // 省略大部分代碼 

本文阿寶哥主要介紹了依賴注入的概念及作用、如何使用 Vue 3 提供的 provide/inject 特性。為了讓大家能夠更深入地理解 provide/inject 特性,阿寶哥從源碼角度分析了 provide 和 inject 函數的具體實現。在后續的文章中,阿寶哥將會介紹在插件中如何應用 provide/inject 特性,感興趣的小伙伴不要錯過喲。

六、參考資源

Vue 3 官網 - Provide/Inject

Vue 3 官網 - 組合式 API

 

責任編輯:武曉燕 來源: 全棧修仙之路
相關推薦

2021-02-26 05:19:20

Vue 3.0 VNode虛擬

2021-02-16 16:41:45

Vue項目指令

2021-02-19 23:07:02

Vue綁定組件

2021-02-22 21:49:33

Vue動態組件

2021-02-18 08:19:21

Vue自定義Vue 3.0

2021-03-04 22:31:02

Vue進階函數

2021-03-09 22:29:46

Vue 響應式API

2021-03-08 00:08:29

Vue應用掛載

2011-03-29 09:51:58

GuiceIOC

2014-01-07 14:53:37

Android開發依賴注入Roboguice

2011-04-15 09:44:45

Spring

2010-05-11 16:22:40

2011-05-31 10:00:21

Android Spring 依賴注入

2023-07-11 09:14:12

Beanquarkus

2017-08-16 16:00:05

PHPcontainer依賴注入

2022-12-29 08:54:53

依賴注入JavaScript

2021-12-15 09:17:12

Spring依賴注入面試題

2020-10-13 08:24:31

Vue3.0系列

2020-04-22 14:15:32

Vue 3.0語法前端

2015-09-02 11:22:36

JavaScript實現思路
點贊
收藏

51CTO技術棧公眾號

欧美电影在线观看完整版| 蜜桃视频在线观看www社区| 18成人免费观看视频| 亚洲女同精品视频| 中文字幕中文在线| 香蕉久久aⅴ一区二区三区| 99re这里只有精品视频首页| 国产福利精品视频| 精品少妇久久久| 欧美人妖在线| 日韩美女视频在线| 老司机午夜av| 888av在线视频| 国产精品―色哟哟| 精品久久精品久久| 一级特黄色大片| 亚洲欧美日韩国产一区二区| 久久视频在线播放| 国产女主播喷水高潮网红在线| 伊人亚洲精品| 欧美视频精品一区| 欧美成人免费在线| 欧亚洲嫩模精品一区三区| 一区二区三区四区视频精品免费| 国产精品专区一| 国产午夜精品无码一区二区| 97视频精品| 日韩精品免费看| 黄页网站在线看| 国产福利亚洲| 一本一本久久a久久精品综合麻豆| 免费观看国产视频在线| gogogo高清在线观看免费完整版| 成人av先锋影音| 91精品婷婷国产综合久久蝌蚪| 一区二区三区麻豆| 性欧美长视频| 97精品欧美一区二区三区| 中国毛片直接看| 日韩精品免费一区二区在线观看| 亚洲欧美999| 无码人妻aⅴ一区二区三区| 2020国产精品极品色在线观看| 在线播放亚洲一区| av污在线观看| 日韩制服诱惑| 欧亚洲嫩模精品一区三区| wwwxxx黄色片| 欧美一区 二区 三区| 狠狠久久亚洲欧美专区| 欧美大片在线播放| 高h视频在线播放| 亚洲一区二三区| 日韩视频 中文字幕| huan性巨大欧美| 亚洲男人天堂av| 视频一区二区视频| av毛片在线免费| 一区二区三区免费| 国产精品www在线观看| 丁香花视频在线观看| 一区二区三区四区不卡在线| 伊人久久婷婷色综合98网| 3p在线观看| 亚洲欧美综合在线精品| 日本特级黄色大片| 影音先锋男人资源在线| 亚洲一区二区三区免费视频| 黄页网站在线观看视频| 欧产日产国产精品视频| 色综合色综合色综合色综合色综合| 欧美性大战久久久久xxx| 偷拍精品精品一区二区三区| 欧美羞羞免费网站| 天天操天天干天天做| 欧美高清hd| 亚洲国产精品美女| 免费人成又黄又爽又色| 爽成人777777婷婷| 欧美成人精品在线| 国产香蕉在线视频| 久久国产88| 国产色婷婷国产综合在线理论片a| 国产一区二区三区视频免费观看| 国产麻豆精品久久一二三| 91亚洲午夜在线| 偷拍精品一区二区三区| 国产日韩欧美一区二区三区乱码| 中文字幕中文字幕一区三区| 色呦呦在线播放| 色欲综合视频天天天| 成人综合久久网| 日韩高清影视在线观看| 中国人与牲禽动交精品| avove在线播放| 销魂美女一区二区三区视频在线| 性一交一乱一区二区洋洋av| 精品91福利视频| 日韩精品dvd| 亚洲视频在线观看网站| 日韩av网站在线播放| 亚洲午夜久久久久久尤物 | 欧美捆绑视频| 成人免费一区二区三区视频| 少妇av一区二区三区无码| 日韩另类视频| 亚洲成人激情在线观看| 亚洲AV成人无码网站天堂久久| 136国产福利精品导航网址| 国产有码在线一区二区视频| 偷拍精品一区二区三区| 亚洲激情校园春色| 九一精品在线观看| 久久97精品| 免费97视频在线精品国自产拍| 日韩在线视频免费播放| 国产精品亚洲午夜一区二区三区| 日本高清不卡一区二区三| 青青在线视频| 在线不卡中文字幕| 成人小视频免费看| 性色一区二区三区| 国产日产精品一区二区三区四区| 精精国产xxxx视频在线| 在线观看免费视频综合| 日本黄色动态图| 亚洲五月婷婷| 97免费高清电视剧观看| 国产在线观看a| 欧美视频日韩视频| 谁有免费的黄色网址| 一区二区国产精品| 国产九色91| 国内在线免费视频| 日韩一区二区在线免费观看| 麻豆一区在线观看| 蜜臀av一区二区三区| 欧美三级网色| 国产精品av一区二区三区| 亚洲国产精品成人一区二区| 久久免费少妇高潮99精品| 国产精品一区二区不卡| 一本二本三本亚洲码| 国产精品日本一区二区不卡视频| 日韩亚洲一区二区| 亚洲天堂自拍偷拍| 国产精品不卡视频| 九九热免费在线观看| 日韩中文在线电影| 成人国产精品免费视频 | 欧美成人在线影院| 99国产揄拍国产精品| 亚洲精品乱码久久久久久| 天天操精品视频| 欧美jizzhd精品欧美巨大免费| 1卡2卡3卡精品视频| a黄色片在线观看| 欧美一区二区三区系列电影| 蜜臀久久精品久久久用户群体| 精品一区二区三区在线观看国产 | www.亚洲视频| 欧美色网站导航| 国产suv精品一区二区68| 国产老妇另类xxxxx| 国产黄色激情视频| 伦理一区二区三区| 国产成人精品亚洲精品| 在线观看免费版| 日韩一区和二区| 国产精品黄色大片| 亚洲国产成人自拍| 波多野结衣免费观看| 激情婷婷久久| 日本一区视频在线| 97色婷婷成人综合在线观看| 九九精品在线观看| 日韩三级电影网| 欧美日韩免费高清一区色橹橹| 爱爱视频免费在线观看| 成人av资源网站| 国产av人人夜夜澡人人爽| 欧美三级伦理在线| 97欧洲一区二区精品免费| 神马午夜在线视频| www.欧美精品| 日本免费网站在线观看| 欧美无乱码久久久免费午夜一区| 欧美色图亚洲视频| 久久久久88色偷偷免费| 亚洲综合123| 亚洲欧美日韩在线观看a三区| 在线国产99| 加勒比色综合久久久久久久久| 国产盗摄xxxx视频xxx69| av在线下载| 亚洲天堂网站在线观看视频| 国产suv一区二区| 日韩欧美a级成人黄色| 欧美成人777| 国产亚洲欧美中文| 中文字幕亚洲日本| 日韩av一区二区在线影视| 欧洲精品在线播放| 四虎成人av| 日本不卡在线播放| 欧美aaaaa级| 99re国产| 亚洲人体在线| 日本亚洲精品在线观看| heyzo高清在线| 久久久av一区| 成黄免费在线| 日韩精品黄色网| 亚洲精品字幕在线观看| 欧美精品三级在线观看| 亚洲国产成人精品女人久久| 亚洲成人激情自拍| 麻豆精品一区二区三区视频| 国产精品视频一二三| 性色av蜜臀av色欲av| 成人午夜精品一区二区三区| 91精产国品一二三产区别沈先生| 免费看的黄色欧美网站 | 奇米在线7777在线精品| 成人黄色片视频| 亚洲精品九九| 国产精品av免费观看| 在线中文字幕亚洲| 在线观看欧美激情| 91亚洲成人| 亚洲一区二区三区精品视频| 狠狠做六月爱婷婷综合aⅴ| 久久亚洲高清| 人人香蕉久久| 久久久久久久久四区三区| aaaa一级片| 精品久久在线观看| 久久久综合精品| 国产麻豆剧传媒精品国产| 久久99精品久久久久久动态图| 色婷婷综合久久久久中文字幕| 亚洲片区在线| 韩日视频在线观看| 欧美午夜在线| 波多野结衣av一区二区全免费观看| 亚洲天天影视网| 九九久久九九久久| 国产精品v亚洲精品v日韩精品 | 日韩精品2区| 亚洲一区二区三区精品视频 | 成人在线观看你懂的| 国内精品久久久久久久影视蜜臀| 欧美黄网在线观看| 伊人狠狠色j香婷婷综合| youjizz.com在线观看| 国产精品久久| 欧美,日韩,国产在线| 亚洲作爱视频| 蜜臀av午夜一区二区三区| 久久资源在线| 日本不卡一区在线| 国产激情视频一区二区在线观看 | 99热这里只有精| 日韩女优电影在线观看| 神马午夜一区二区| 亚洲欧美国产制服动漫| 2021av在线| 九九久久国产精品| 1234区中文字幕在线观看| 日本在线观看天堂男亚洲| 国产精品第一| 51国偷自产一区二区三区的来源| 国产无遮挡裸体免费久久| 欧美精品七区| 香蕉精品视频在线观看| 日本欧美视频在线观看| 视频一区二区三区在线| 三日本三级少妇三级99| 99国产精品久| 91ts人妖另类精品系列| 一区二区三区在线观看动漫| 久久久久久久久久91| 色88888久久久久久影院按摩| 一级日韩一级欧美| 日韩精品欧美国产精品忘忧草| 91吃瓜网在线观看| 欧美国产日本在线| 刘亦菲一区二区三区免费看| 91免费欧美精品| 牲欧美videos精品| 国产精品亚洲天堂| 在线综合亚洲| 日韩久久久久久久久久久| www久久精品| 免费看一级大片| 色悠悠久久综合| 精品人妻少妇嫩草av无码专区| 亚洲欧美色婷婷| 亚洲综合影视| 国产精品一区二区三区在线播放 | 国产精品第9页| 555www色欧美视频| 精彩国产在线| 久久久免费精品| 亚洲精品777| 日韩精品最新在线观看| 亚洲精品专区| 国产资源中文字幕| 欧美国产精品中文字幕| www.日本精品| 精品国产伦一区二区三区观看体验| 97超碰人人在线| 51精品在线观看| 国产精品中文字幕制服诱惑| eeuss中文| 久久er99精品| av手机在线播放| 欧美日韩亚洲视频| 成人久久精品人妻一区二区三区| www.久久久久| 欧美黄色a视频| 日韩免费中文专区| 午夜一区不卡| 男人的天堂影院| 亚洲无人区一区| 精品区在线观看| 久久电影一区二区| 24小时成人在线视频| 亚洲高清精品中出| 久久精品麻豆| 亚洲最大成人网站| 黑人巨大精品欧美一区二区| 亚洲精品成人区在线观看| 欧美精品在线网站| www.久久99| 992tv成人免费观看| 久久69国产一区二区蜜臀| 亚洲AV无码成人精品区明星换面| 欧美视频中文字幕在线| 丝袜视频国产在线播放| 欧美又大又粗又长| 一区二区美女| 男人女人黄一级| 国产人伦精品一区二区| 69av视频在线观看| 在线观看日韩视频| 国产精品.xx视频.xxtv| 伊人精品久久久久7777| 狠狠色丁香婷婷综合久久片| 欧美一级片在线视频| 欧美一区二区在线观看| 五月天激情在线| 国产免费一区| 国产精品腿扒开做爽爽爽挤奶网站| www.久久国产| 欧美在线观看视频一区二区三区| av中文字幕在线| 91精品视频在线| 国内综合精品午夜久久资源| 年下总裁被打光屁股sp | 精品国产导航| 国产午夜福利视频在线观看| 国产肉丝袜一区二区| 一级片aaaa| 久久久久久尹人网香蕉| 亚州国产精品| 日韩不卡一二三| 一区二区三区不卡在线观看| 性生活免费网站| 日本精品视频在线观看| 成人一二三区| 97免费公开视频| 黑人巨大精品欧美一区二区免费| 国产成人天天5g影院在线观看| 成人黄色av网站| 亚洲三级国产| 综合 欧美 亚洲日本| 日韩精品专区在线影院重磅| 人人草在线视频| 在线观看精品视频| av在线这里只有精品| 国产成人自拍偷拍| 久久成人人人人精品欧| 日本一道高清一区二区三区| 亚洲污视频在线观看| 亚洲国产一区二区视频| 电影在线一区| 国产精品日韩高清| 日本不卡免费在线视频| 欧美日韩精品亚洲精品| 精品一区二区亚洲| 国产精品一区二区三区www| 看av免费毛片手机播放| 成人欧美一区二区三区白人 | 91女厕偷拍女厕偷拍高清| 日本成人一级片| 97人人做人人爱| 天天av综合| 免费观看a级片| 亚洲精品一区二区三区在线观看|