国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

從Java走進Scala:包和訪問修飾符

開發 后端
在現實生活中,代碼一定要引用并打包,在本文中,Ted Neward 介紹了 Scala 的包(package)和訪問修飾符功能,糾正了以前的疏忽。然后,他繼續探討了 Scale 中的函數內容:“apply” 機制。

最近,讀者的反饋讓我意識到在制作本系列的過程中我遺漏了 Scala 的語言的一個重要方面:Scala 的包和訪問修飾符功能。所以在研究該語言的函數性元素 apply 機制前,我將先介紹包和訪問修飾符。

打包

為了有助于隔離代碼,使其不會相互沖突,Java 代碼提供了 package 關鍵詞,由此創建了一個詞法命名空間,用以聲明類。本質上,將類 Foo 放置到名為 com.tedneward.util 包中就將正式類名修改成了 com.tedneward.util.Foo;同理,必須按該方法引用類。如果沒有,Java 編程人員會很快指出,他們會 import 該包,避免鍵入正式名的麻煩。的確如此,但這僅意味著根據正式名引用類的工作由編譯器和字節碼完成。快速瀏覽一下 javap 的輸出,這點就會很明了。

然而,Java 語言中的包還有幾個特殊的要求:一定要在包所作用的類所在的 .java 文件的頂端聲明包(在將注釋應用于包時,這一點會引發很嚴重的語言問題);該聲明的作用域為整個文件。這意味著兩個跨包進行緊密耦合的類一定要在跨文件時分離,這會致使兩者間的緊密耦合很容易被忽略。

Scala 在打包方面所采取的方法有些不同,它結合使用了 Java 語言的 declaration 方法和 C# 的 scope(限定作用域)方法。了解了這一點,Java 開發人員就可以使用傳統的 Java 方法并將 package 聲明放在 .scala 文件的頂部,就像普通的 Java 類一樣;包聲明的作用域為整個文件,就像在 Java 代碼中一樣。而 Scala 開發人員則可以使用 Scala 的包 “(scoping)限定作用域” 方法,用大括號限制 package 語句的作用域,如清單 1 所示:

清單 1. 簡化的打包

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.       package demonstration  
  8.       {  
  9.         object App  
  10.         {  
  11.           def main(args : Array[String]) : Unit =  
  12.           {  
  13.             System.out.println("Howdy, from packaged code!")  
  14.             args.foreach((i) => System.out.println("Got " + i) )  
  15.           }  
  16.         }  
  17.       }  
  18.     }  
  19.   }  

這個代碼有效地聲明了類 App,或者更確切的說是一個稱為 com.tedneward.scala.demonstration.App 的單個類。注意 Scala 還允許用點分隔包名,所以清單 1 中的代碼可以更簡潔,如清單 2 所示:

清單 2. 簡化了的打包(redux)

  1. package com.tedneward.scala.demonstration  
  2. {  
  3.   object App  
  4.   {  
  5.       def main(args : Array[String]) : Unit =  
  6.       {  
  7.         System.out.println("Howdy, from packaged code!")  
  8.         args.foreach((i) => System.out.println("Got " + i) )  
  9.       }  
  10.   }  

用哪一種樣式看起來都比較合適,因為它們都編譯出一樣的代碼構造(Scala 將繼續編譯并和 javac 一樣在聲明包的子目錄中生成 .class 文件)。

#p#

導入

與包相對的當然就是 import 了,Scala 使用它將名稱放入當前詞法名稱空間。本系列的讀者已經在此前的很多例子中見到過 import 了,但現在我將指出一些讓 Java 開發人員大吃一驚的 import 的特性。

首先,import 可以用于客戶機 Scala 文件內的任何地方,并非只可以用在文件的頂部,這樣就有了作用域的關聯性。因此,在清單 3 中,java.math.BigInteger 導入的作用域被完全限定到了在 App 對象內部定義的方法,其他地方都不行。如果 mathfun 內的其他類或對象要想使用 java.math.BigInteger,就需要像 App 一樣導入該類。如果 mathfun 的幾個類都想使用 java.math.BigInteger,可以在 App 的定義以外的包級別導入該類,這樣在包作用域內的所有類就都導入 BigInteger 了。

清單 3. 導入的作用域

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.         // ...  
  8.         
  9.       package mathfun  
  10.       {  
  11.         object App  
  12.         {  
  13.           import java.math.BigInteger  
  14.           
  15.           def factorial(arg : BigInteger) : BigInteger =  
  16.           {  
  17.             if (arg == BigInteger.ZERO) BigInteger.ONE  
  18.             else arg multiply (factorial (arg subtract BigInteger.ONE))  
  19.           }  
  20.           
  21.           def main(args : Array[String]) : Unit =  
  22.           {  
  23.             if (args.length > 0)  
  24.               System.out.println("factorial " + args(0) +  
  25.                 " = " + factorial(new BigInteger(args(0))))  
  26.             else 
  27.               System.out.println("factorial 0 = 1")  
  28.           }  
  29.         }  
  30.       }  
  31.     }  
  32.   }  

不只如此,Scala 還不區分高層成員和嵌套成員,所以您不僅可以使用 import 將嵌套類型的成員置于詞法作用域中,其他任何成員均可;例如,您可以通過導入 java.math.BigInteger 內的所有名稱,使對 ZERO 和 ONE 的限定了作用域的引用縮小為清單 4 中的名稱引用:

清單 4. 靜態導入

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.         // ...  
  8.    
  9.       package mathfun  
  10.       {  
  11.         object App  
  12.         {  
  13.           import java.math.BigInteger  
  14.           import BigInteger._  
  15.           
  16.           def factorial(arg : BigInteger) : BigInteger =  
  17.           {  
  18.             if (arg == ZERO) ONE  
  19.             else arg multiply (factorial (arg subtract ONE))  
  20.           }  
  21.           
  22.           def main(args : Array[String]) : Unit =  
  23.           {  
  24.             if (args.length > 0)  
  25.               System.out.println("factorial " + args(0) +  
  26.                 " = " + factorial(new BigInteger(args(0))))  
  27.             else 
  28.               System.out.println("factorial 0 = 1")  
  29.           }  
  30.         }  
  31.       }  
  32.     }  
  33.   }  

您可以使用下劃線(還記得 Scala 中的通配符吧?)有效地告知 Scala 編譯器 BigInteger 內的所有成員都需要置入作用域。由于 BigInteger 已經被先前的導入語句導入到作用域中,因此無需顯式地使用包名限定類名。實際上,可以將所有這些都結合到一個語句中,因為 import 可以同時導入多個目標,目標間用逗號隔開(如清單 5 所示):

清單 5. 批量導入

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.         // ...  
  8.    
  9.       package mathfun  
  10.       {  
  11.         object App  
  12.         {  
  13.           import java.math.BigInteger, BigInteger._  
  14.           
  15.           def factorial(arg : BigInteger) : BigInteger =  
  16.           {  
  17.             if (arg == ZERO) ONE  
  18.             else arg multiply (factorial (arg subtract ONE))  
  19.           }  
  20.           
  21.           def main(args : Array[String]) : Unit =  
  22.           {  
  23.             if (args.length > 0)  
  24.               System.out.println("factorial " + args(0) +  
  25.                 " = " + factorial(new BigInteger(args(0))))  
  26.             else 
  27.               System.out.println("factorial 0 = 1")  
  28.           }  
  29.         }  
  30.       }  
  31.     }  
  32.   }  

這樣您可以節省一兩行代碼。注意這兩個導入過程不能結合:先導入 BigInteger 類本身,再導入該類中的各種成員。

也可以使用 import 來引入其他非常量的成員。例如,考慮一下清單 6 中的數學工具庫(或許不一定有什么價值):

清單 6. Enron 的記帳代碼

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.         // ...  
  8.         
  9.       package mathfun  
  10.       {  
  11.         object BizarroMath  
  12.         {  
  13.           def bizplus(a : Int, b : Int) = { a - b }  
  14.           def bizminus(a : Int, b : Int) = { a + b }  
  15.           def bizmultiply(a : Int, b : Int) = { a / b }  
  16.           def bizdivide(a : Int, b : Int) = { a * b }  
  17.         }  
  18.       }  
  19.     }  
  20.   }  

使用這個庫會越來越覺得麻煩,因為每請求它的一個成員,都需要鍵入 BizarroMath,但是 Scala 允許將 BizarroMath 的每一個成員導入最高層的詞法空間,因此簡直就可以把它們當成全局函數來使用(如清單 7所示):

清單 7. 計算 Enron的開支

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.       package demonstration  
  8.       {  
  9.         object App2  
  10.         {  
  11.           def main(args : Array[String]) : Unit =  
  12.           {  
  13.             import com.tedneward.scala.mathfun.BizarroMath._  
  14.               
  15.             System.out.println("2 + 2 = " + bizplus(2,2))  
  16.           }  
  17.         }  
  18.       }  
  19.     }  
  20.   }  

還有其他的一些構造很有趣,它們允許 Scala 開發人員寫出更自然的 2 bizplus 2,但是這些內容本文不予討論(想了解 Scala 潛在的可以用于其他用途的特性的讀者可以看一下 Odersky、Spoon 和 Venners 所著的 Programming in Scala 中談到的 Scala implicit 構造)。

#p#

訪問

打包(和導入)是 Scala 封裝的一部分,和在 Java 代碼中一樣,在 Scala 中,打包很大一部分在于以選擇性方式限定訪問特定成員的能力 — 換句話說,在于 Scala 將特定成員標記為 “公有(public)”、“private(私有)” 或介于兩者之間的成員的能力。

Java 語言有四個級別的訪問:公有(public)、私有(private)、受保護的(protected )和包級別(它沒有任何關鍵詞)訪問。Scala:

廢除了包級別的限制(在某種程度上)

默認使用 “公有”

指定 “私有” 表示 “只有此作用域可訪問”

相反,Scala 定義 “protected” 的方式與在 Java 代碼中不同;Java protected 成員對于子類和在其中定義成員的包來說是可訪問的,Scala 中則僅有子類可訪問。這意味著 Scala 版本的 protected 限制性要比 Java 版本更嚴格(雖然按理說更加直觀)。

然而,Scala 真正區別于 Java 代碼的地方是 Scala 中的訪問修飾符可以用包名來 “限定”,用以表明直到 哪個訪問級別才可以訪問成員。例如,如果 BizarroMath 包要將成員訪問權限授權給同一包中的其他成員(但不包括子類),可以用清單 8 中的代碼來實現:

清單 8. Enron 的記帳代碼

  1. package com  
  2. {  
  3.   package tedneward  
  4.   {  
  5.     package scala  
  6.     {  
  7.         // ...  
  8.         
  9.       package mathfun  
  10.       {  
  11.         object BizarroMath  
  12.         {  
  13.           def bizplus(a : Int, b : Int) = { a - b }  
  14.           def bizminus(a : Int, b : Int) = { a + b }  
  15.           def bizmultiply(a : Int, b : Int) = { a / b }  
  16.           def bizdivide(a : Int, b : Int) = { a * b }  
  17.       
  18.               private[mathfun] def bizexp(a : Int, b: Int) = 0 
  19.         }  
  20.       }  
  21.     }  
  22.   }  

注意此處的 private[mathfun] 表達。本質上,這里的訪問修飾符是說該成員直到 包 mathfun 為止都是私有的;這意味著包 mathfun 的任何成員都有權訪問 bizexp,但任何包以外的成員都無權訪問它,包括子類。

這一點的強大意義就在于任何包都可以使用 “private” 或者 “protected” 聲明甚至 com(乃至 _root_,它是根名稱空間的別名,因此本質上 private[_root_] 等效于 “public” 同)進行聲明。這使得 Scala 能夠為訪問規范提供一定程度的靈活性,遠遠高于 Java 語言所提供的靈活性。

實際上,Scala 提供了一個更高程度的訪問規范:對象私有 規范,用 private[this] 表示,它規定只有被同一對象調用的成員可以訪問有關成員,其他對象里的成員都不可以,即使對象的類型相同(這彌合了 Java 訪問規范系統中的一個缺口,這個缺口除對 Java 編程問題有用外,別無他用。)

注意訪問修飾符必須在某種程度上在 JVM 之上映射,這致使定義中的細枝末節會在從正規 Java 代碼中調用或編譯時丟失。例如,上面的 BizarroMath 示例(用 private[mathfun] 聲明的成員 bizexp)將會生成清單 9 中的類定義(當用 javap 來查看時):

Listing 9. Enron 的記帳庫,JVM 視圖

  1. Compiled from "packaging.scala" 
  2. public final class com.tedneward.scala.mathfun.BizarroMath  
  3.    extends java.lang.Object  
  4. {  
  5.     public static final int $tag();  
  6.     public static final int bizexp(intint);  
  7.     public static final int bizdivide(intint);  
  8.     public static final int bizmultiply(intint);  
  9.     public static final int bizminus(intint);  
  10.     public static final int bizplus(intint);  

在編譯的 BizarroMath 類的第二行很容易看出,bizexp() 方法被賦予了 JVM 級別的 public 訪問修飾符,這意味著一旦 Scala 編譯器結束訪問檢查,細微的 private[mathfun] 區別就會丟失。因此,對于那些要從 Java 代碼使用的 Scala 代碼,我寧愿堅持傳統的 “private” 和 “public” 的定義(甚至 “protected” 的定義有時最終映射到 JVM 級別的 “public”,所有不確定的時候,請對照實際編譯的字節碼參考一下 javap,以確認其訪問級別。)

#p#

應用

在本系列上一期的文章中(“集合類型”),當談及 Scala 中的數組時(確切地說是 Array[T])我說過:“獲取數組的第 i 個元素” 實際上是 “那些名稱很有趣的方法中的一種……”。盡管當時是因為我不想深入細節,但不管怎么說事實證明這種說法嚴格來說 是不對的。

好吧,我承認,我說謊了。

技術上講,在 Array[T] 類上使用圓括號要比使用 “名稱有趣的方法” 復雜一點;Scala 為特殊的字符序列(即那些有左右括號的序列)保留了一個特殊名稱關聯,因為它有著特殊的使用意圖 :“做”……(或按函數來說,將……“應用” 到……)。

換句話說,Scala 有一個特殊的語法(更確切一些,是一個特殊的語法關系)來代替 “應用” 操作符 “()”。更精確地說,當用 () 作為方法調用來調用所述對象時,Scala 將稱為 apply() 的方法作為調用的方法。例如,一個想充當仿函數(functor)的類(一個充當函數的對象)可以定義一個 apply 方法來提供類似于函數或方法的語義:

清單 10. 使用 Functor!

  1. class ApplyTest  
  2. {  
  3.   import org.junit._, Assert._    
  4.     
  5.   @Test def simpleApply =  
  6.   {  
  7.     class Functor  
  8.     {  
  9.       def apply() : String =  
  10.       {  
  11.         "Doing something without arguments" 
  12.       }  
  13.         
  14.       def apply(i : Int) : String =  
  15.       {  
  16.         if (i == 0)  
  17.           "Done" 
  18.         else 
  19.           "Applying... " + apply(i - 1)  
  20.       }  
  21.     }  
  22.  
  23.     val f = new Functor  
  24.     assertEquals("Doing something without arguments", f() )  
  25.     assertEquals("Applying... Applying... Applying... Done", f(3))  
  26.   }  
  27. }  

好奇的讀者會想是什么使仿函數不同于匿名函數或閉包呢?事實證明,它們之間的關系相當明顯:標準 Scala 庫中的 Function1 類型(指包含一個參數的函數)在其定義上有一個 apply 方法。快速瀏覽一些為 Scala 匿名函數生成的 Scala 匿名類,您就會明白生成的類是 Function1(或者 Function2 或 Function3,這要看該函數使用了幾個參數)的后代。

這意味著當匿名的或者命名的函數不一定適合期望設計方法時,Scala 開發人員可以創建一個 functor 類,提供給它一些初始化數據,保存在字段中,然后通過 () 執行它,無需任何通用基類(傳統的策略模式實現需要這個類):

清單 11. 使用 Functor!

  1. class ApplyTest  
  2. {  
  3.   import org.junit._, Assert._    
  4.  
  5.   // ...  
  6.     
  7.   @Test def functorStrategy =  
  8.   {  
  9.     class GoodAdder  
  10.     {  
  11.       def apply(lhs : Int, rhs : Int) : Int = lhs + rhs  
  12.     }  
  13.     class BadAdder(inflateResults : Int)  
  14.     {  
  15.       def apply(lhs : Int, rhs : Int) : Int = lhs + rhs * inflateResults  
  16.     }  
  17.  
  18.     val calculator = new GoodAdder  
  19.     assertEquals(4, calculator(22))  
  20.     val enronAccountant = new BadAdder(50)  
  21.     assertEquals(102, enronAccountant(22))  
  22.   }  
  23. }  

任何提供了被適當賦予了參數的 apply 方法的類,只要這些參數都按數字和類型排列了起來,它們都會在被調用時運行。

結束語

Scala 的打包、導入和訪問修飾符機制提供了傳統 Java 編程人員從未享受過的更高級的控制和封裝。例如,它們提供了導入一個對象的選擇方法的能力,使它們看起來就像全局方法一樣,而且還克服了全局方法的傳統的缺點;它們使得使用那些方法變得極其簡單,尤其是當這些方法提供了諸如本系列早期文章(“Scala 控制結構內部揭密”)引入的虛構的 tryWithLogging 函數這樣的高級功能時。

同樣,“應用” 機制允許 Scala 隱藏函數部分的執行細節,這樣,編程人員可能會不知道(或不在乎)他們正調用的東西 事實上不是一個函數,而是一個非常復雜的對象。該機制為 Scala 機制的函數特性提供了另一個方面,當然 Java 語言(或者 C# 或 C++)也提供了這個方面,但是它們提供的語法純度沒有 Scala 的高。

【相關閱讀】

  1. Scala編程語言專題
  2. 面向Java開發人員的Scala指南:使用元組、數組和列表
  3. 面向Java開發人員的Scala指南:當繼承中的對象遇到函數
  4. 面向Java開發人員的Scala指南:使用Scala版本的Java接口
  5. 面向Java開發人員的Scala指南:Scala控制結構內部揭密
責任編輯:yangsai 來源: IBMDW
相關推薦

2009-08-24 16:49:39

C#修飾符

2009-06-12 13:37:47

訪問權限修飾符Java教程

2009-09-04 11:06:40

C#訪問修飾符

2020-09-02 07:03:04

虛擬機HotSpotJava

2009-09-28 11:01:39

從Java走進Scal

2009-08-21 16:17:25

ScalaTwitter API

2009-07-22 08:45:35

Scala超類構造器override修飾符

2009-06-16 17:54:38

Scala類語法語義

2011-06-02 14:51:07

JAVA修飾符

2009-06-17 13:57:25

Scala元組數組

2009-06-17 11:44:22

Scala控制結構

2015-08-18 09:25:11

Java修飾符關鍵詞

2009-12-09 09:15:47

從Java走進ScalTwitter API

2009-07-15 10:14:25

Scala并發性

2009-02-04 17:32:03

ibmdwJavaScala

2023-12-29 09:01:27

SwiftUI視圖修飾符

2009-08-27 11:04:08

C# extern修飾

2009-08-27 13:06:13

C# new修飾符

2009-09-02 17:14:28

C#修飾符

2025-08-15 11:10:54

點贊
收藏

51CTO技術棧公眾號

国产精品传媒| 不卡视频一区二区三区| 久久91成人| 色94色欧美sute亚洲线路一ni| 亚洲成人动漫在线| 亚洲欧美一区在线| 欧美精品激情blacked18| 蜜桃成人365av| 亚洲一本大道在线| 99久久国产免费看| 色综合色狠狠综合色| 成年人视频网站免费观看| 欧美深夜福利| 久久久久久亚洲精品| 免费污视频在线| 欧美日韩精品国产| 天天视频天天爽| 国产成人精品一区二区三区四区| 97伦理在线四区| 免费看成人哺乳视频网站| 色妞欧美日韩在线| 免费毛片b在线观看| 欧美三级午夜理伦三级中视频| 久草福利资源站| 成人av网站免费| 精品日韩在线播放| 视频一区中文字幕| 成人久久18免费网站漫画| 亚洲午夜久久| 久久久噜久噜久久综合| 国产69精品久久| 日韩av影院在线观看| 丁香婷婷在线观看| 午夜精品久久久久久久99樱桃| 亚洲五月天综合| 91在线你懂得| 波多野结衣乳巨码无在线| 国产激情视频一区二区在线观看 | 欧美午夜无遮挡| 色总=综合色| 国产亚洲视频系列| 少妇高潮喷水久久久久久久久久| 国产精品一区二区三区四区| 热久久最新地址| 国产 日韩 欧美大片| 午夜啪啪福利视频| 福利电影一区二区三区| 91传媒免费视频| 91久久极品少妇xxxxⅹ软件| re久久精品视频| 国产成人在线视频| 欧美日韩水蜜桃| 91九色在线视频| 在线精品亚洲| 亚洲精品国产一区| 国产白丝精品91爽爽久久| 青草青青在线视频| 国产亚洲综合色| 97在线资源在| 亚洲电影第三页| 欧美少妇另类| 欧美一区二区三区在线| 青春草免费在线视频| 亚洲第一综合天堂另类专| 漫画在线观看av| 色噜噜狠狠色综合网图区 | 欧美性生交大片免费| 国产视频二区在线观看| 欧美日本免费一区二区三区| 2018av在线| 欧美成人午夜视频| 成人3d动漫在线观看| 久久综合九色99| 国产不卡在线一区| 天堂中文av| 欧美一二三四在线| 国模大尺度视频一区二区| 国产精品狼人色视频一区| 鲁大师成人一区二区三区| 欧美极品欧美精品欧美| 五月激情综合婷婷| av剧情在线观看| 欧美在线欧美在线| 好看的日韩av电影| 日本一区午夜艳熟免费| 亚洲午夜久久久久中文字幕久| 免费黄色网址在线观看| 久久精品2019中文字幕| 99免费精品| 夜夜爽www精品| 亚洲精品视频一区二区| 最爽无遮挡行房视频在线| 欧美成人免费小视频| 中国成人一区| 国产黄色激情视频| 亚洲va欧美va国产va天堂影院| aa级大片免费在线观看| 青青草一区二区| 精品亚洲aⅴ乱码一区二区三区| 免费高清成人| 日韩女优制服丝袜电影| 偷拍精品福利视频导航| 亚洲国产综合自拍| 亚洲影视在线观看| 伊人久久综合一区二区| 国产视频999| 91免费视频网址| 最新超碰在线| 国产精品人成电影在线观看| 国产一区999| 免费观看成年在线视频网站| 久久精品国产一区| 久久九九电影| 自拍偷拍电影| 国产亚洲欧美日韩精品| 亚洲网址在线| 69国产精品视频| www国产精品com| 欧美aaaaaa午夜精品| 在线观看国产高清视频| 欧美精品制服第一页| 黄色亚洲在线| 91大神在线资源观看无广告| 中文字幕日韩欧美| 日韩精品一级二级 | 国内精品久久久久影院薰衣草| 女人体1963| 国产亚洲欧洲在线| 亚洲精品免费在线观看| 欧美成熟毛茸茸| 亚洲国产欧美在线成人app| 日本一区二区三区中文字幕| 国产免费一区二区三区| 国产成人在线免费| 最新国产在线观看| 深夜成人在线观看| 国产精品1luya在线播放| 国产精品视频免费在线| 日韩电影在线一区| 久久99国产精品| 成人看av片| 国产精品久久久久久久久久久久| 精品一区二区三区不卡| 久久99久久| 欧美有码在线视频| 久久综合久久综合九色| 欧美xoxoxo| 亚洲高清在线观看一区| 欧美日韩黄色一区二区| 国产精品99一区二区三| 国产成免费视频| 久久免费少妇高潮久久精品99| 精品亚洲国产成人av制服丝袜| 黄色小网站在线观看| 91成人伦理在线电影| 亚洲精品午夜久久久| 牛牛精品成人免费视频| 99免费视频| 欧美专区中文字幕| 亚洲婷婷国产精品电影人久久| av成人综合| 成人网18免费看| 午夜精品在线视频| 国产欧美视频一区二区三区| 久久久久久久久久久久电影| 欧美日本视频在线观看| 久久视频免费观看| 国产人成一区二区三区影院| 一区二区免费| av在线dvd| 国产在线观看精品一区二区三区| 亚洲国产日韩一级| 91中文字幕精品永久在线| 国产精品玖玖玖在线资源| 色乱码一区二区三区网站| 91福利免费在线| 国产aⅴ夜夜欢一区二区三区 | 精品中国亚洲| 超碰超碰在线观看| 日本成人精品在线| 亚洲国产精品嫩草影院| 色999国产精品| 国产视频网站在线| 欧美亚洲另类久久综合| 亚洲高清免费观看高清完整版| 狠狠色丁香婷综合久久| 全球最大av网站久久| 爱情岛论坛成人| 国产91色在线|免| 99热一区二区| 7777精品伊人久久久大香线蕉完整版| 国产在线日韩精品| 欧美日韩视频一区二区三区| 在线日韩一区二区| 日韩精品另类天天更新| 1024国产在线| 好吊视频一区二区三区四区| 欧美日韩国内自拍| 亚洲人成五月天| 国产另类自拍|