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

你可能不知道的陷阱, IEnumerable接口

開發(fā) 后端
IEnumerable枚舉器接口的重要性,說一萬句話都不過分。幾乎所有集合都實現(xiàn)了這個接口,Linq的核心也依賴于這個萬能的接口。C語言的for循環(huán)寫得心煩,foreach就順暢了很多。

IEnumerable枚舉器接口的重要性,說一萬句話都不過分。幾乎所有集合都實現(xiàn)了這個接口,Linq的核心也依賴于這個***的接口。C語言的for循環(huán)寫得心煩,foreach就順暢了很多。

我很喜歡這個接口,但在使用中也遇到不少的疑問,你是不是也有與我一樣的困惑:

(1) IEnumerable 與  IEnumerator到底有什么區(qū)別

(2) 枚舉能否越界訪問,越界訪問是什么后果?為什么在枚舉中不能改變集合的值?

(3) Linq的具體實現(xiàn)到底是怎樣的,比如Skip,它跳過了一些元素,那么這些元素被訪問到了么?

(4) IEnumerable 的本質(zhì)是什么?

(5) IEnumerable 枚舉中是否會形成閉包?多個枚舉過程會不會互相干擾?能否在枚舉中動態(tài)改變枚舉的元素?

....

如果感興趣,我們接著下面的內(nèi)容。

開始之前,我們的文章規(guī)定,枚舉就是IEnumerable,迭代就是IEnumerator,已經(jīng)被實例化(比如ToList())就是集合。

我的相似的一篇博文:你可能不知道的陷阱:C#委托和事件的困惑

1.  IEnumerable 與  IEnumerator

IEnumerable只有一個抽象方法:GetEnumerator(),而IEnumerator又是一個迭代器,真正實現(xiàn)了訪問集合的功能。  IEnumerator只有一個Current屬性,MoveNext和Reset兩個方法。

有個小問題,只搞一個訪問器接口不就得了?為什么要兩個看起來很容易混淆的接口呢?一個叫枚舉器,另一個叫迭代器。因為

(1) 實現(xiàn)IEnumerator是個臟活累活,白白加了兩個方法一個屬性,而且這兩個方法其實并不好實現(xiàn)(后面會提到)。

(2) 它需要維護初始狀態(tài),知道如何MoveNext ,如何結(jié)束,同時返回迭代的上一個狀態(tài),這些并不容易。

(3)迭代顯然是非線程安全的,每次IEnumerable都會生成新的IEnumerator,從而形成多個互相不影響的迭代過程。在迭代過程中,不能修改迭代集合,否則不安全。

所以只要你實現(xiàn)了IEnumerable,編譯器就會幫我們實現(xiàn)IEnumerator。何況絕大多數(shù)情況都是從現(xiàn)有集合繼承,一般不需要重寫MoveNext和Reset方法。 IEnumerable當然還有泛型實現(xiàn),這個不影響問題的討論。

IEnumerable讓我們想起了單向鏈表,C中需要一個指針域保存下一個節(jié)點的信息,那么在IEnumerable中,誰幫忙保存了這個信息?這個過程占用內(nèi)存么? 是占在程序區(qū),還是堆區(qū)?

但是,IEnumerable也有它的缺點,它沒法后退,沒法跳躍(只能一個一個的跳過去),而且實現(xiàn)Reset并不容易,無法實現(xiàn)索引訪問。想想看, 如果是一個實例集合的枚舉過程,直接返回到第0個元素就可以了,但是如果這個IEnumerable是漫長的訪問鏈條,想找到最初的根是很困難的!所 以CLR via C#的作者告訴你,其實很多Reset的實現(xiàn)根本就是謊言,知道有這個東西就行了,不要太過依賴它。

2. foreach和MoveNext有區(qū)別嗎

IEnumerable***的特點是將訪問的過程,交給了被訪問者本身控制。在C語言中數(shù)組控制權(quán)是外部完全掌握的。這個接口卻在內(nèi)部封裝訪問了的過程,進一步提升了封裝性。比如下面

  1. public class People  //定義一個簡單的實體類 
  2.     { 
  3.         public string Name { getset; } 
  4.         public int Age { getset; } 
  5.     } 
  6.  
  7.     public class PersonList 
  8.     { 
  9.         private readonly List<People> peoples; 
  10.  
  11.         public PersonList()  //為了方便,構(gòu)造過程中插入元素 
  12.         { 
  13.             peoples = new List<People>(); 
  14.             for (int i = 0; i < 5; i++) 
  15.             { 
  16.                 peoples.Add(new People {Name = "P" + i, Age = 30 + i}); 
  17.             } 
  18.         } 
  19.  
  20.         public int OldAge = 31; 
  21.         public IEnumerable<People> OlderPeoples 
  22.         { 
  23.             get 
  24.             { 
  25.                 foreach (People people in _people) 
  26.                 { 
  27.                     if (people.Age > OldAge) 
  28.                         yield return people; 
  29.                 } 
  30.                 yield break
  31.             } 
  32.         } 
  33.     } 

IEnumerable的本質(zhì)是狀態(tài)機,它有點類似事件的概念,將實現(xiàn)丟到外面,實現(xiàn)代碼間的穿越(想想星際穿越),這是Linq的基礎(chǔ)。酷炫的迭代器,真的有我們想象的那么簡單么?

在C語言中,數(shù)組就是數(shù)組,實實在在的內(nèi)存空間,那么IEnumerable到底是什么意思呢?如果它由一個真正的集合(比如List)實現(xiàn),那么沒問題,也是實實在在的內(nèi)存,可是如果是上述的例子呢?篩選返回的yield return 只返回了元素,但可能并不存在這個實際的集合,如果你將簡單的枚舉器的yield return 反編譯后看,會發(fā)現(xiàn)其實是一組switch-case, 編譯器在后臺為我們做了大量的工作。

生成的新迭代器,如果不MoveNext,其實Current是空的,這是為什么呢?為什么一個迭代器不直接指向頭元素呢?

(感謝回答:就像C語言的單向鏈表的頭指針一樣,這樣可以指定一個不包含任何元素的枚舉,程序設(shè)計起來更方便)

foreach每次往前移動一格,到頭了就停止。 等等,你確定它到頭了就會停止么?我們來做個試驗:

  1. public IEnumerable<People> Peoples1   //直接返回集合 
  2.         { 
  3.             get { return peoples; } 
  4.         }public IEnumerable<People> Peoples2  //包含yield break; 
  5.         { 
  6.             get 
  7.             { 
  8.                 foreach (var people in peoples) 
  9.                 { 
  10.                     yield return people; 
  11.                 } 
  12.                 yield break;  //其實這個用不用都可以 
  13.             } 
  14.         } 

以上兩種,是我們常見的方式,注意第二種實現(xiàn),ReSharper把yield break標成灰色(重復(fù))。

我們再寫下如下的測試代碼,peopleList集合只有五個元素,但嘗試去MoveNext 8次。可以把peopleList.Peoples1換成2,3,分別測試。

  1. var peopleList = new PeopleList();  //內(nèi)部構(gòu)造函數(shù)插入了五個元素 
  2.             IEnumerator<People> e1 = peopleList.Peoples1.GetEnumerator(); 
  3.             if (e1.Current == null
  4.             { 
  5.                 Console.WriteLine("迭代器生成后Current為空"); 
  6.             } 
  7.             int i = 0; 
  8.             while (i<8)  //總共只有五個元素,看看一直迭代會發(fā)生什么效果 
  9.             { 
  10.                 e1.MoveNext(); 
  11.                 if (e1.Current == null
  12.                 { 
  13.                     Console.WriteLine("迭代第{0}次后為空",i); 
  14.                 } 
  15.                 else 
  16.                 { 
  17.                     Console.WriteLine("迭代第{0}次后為{1}",i,e1.Current.Name); 
  18.                 } 
  19.                 i++; 
  20.             } 

越界枚舉測試結(jié)果

  1. //PeopleEnumerable1   (直接返回集合) 
  2. 迭代器生成后Current為空 
  3. 迭代第0次后為P0 
  4. 迭代第1次后為P1 
  5. 迭代第2次后為P2 
  6. 迭代第3次后為P3 
  7. 迭代第4次后為P4 
  8. 迭代第5次后為空 
  9. 迭代第6次后為空 
  10. 迭代第7次后為空 
  11.  
  12. //PeopleEnumerable2 (不加yield break) 
  13. 迭代器生成后Current為空 
  14. 迭代第0次后為P0 
  15. 迭代第1次后為P1 
  16. 迭代第2次后為P2 
  17. 迭代第3次后為P3 
  18. 迭代第4次后為P4 
  19. 迭代第5次后為P4 
  20. 迭代第6次后為P4 
  21. 迭代第7次后為P4 
  22.  
  23.  
  24. //PeopleEnumerable2 (加上yield break) 
  25. 迭代器生成后Current為空 
  26. 迭代第0次后為P0 
  27. 迭代第1次后為P1 
  28. 迭代第2次后為P2 
  29. 迭代第3次后為P3 
  30. 迭代第4次后為P4 
  31. 迭代第5次后為P4 
  32. 迭代第6次后為P4 
  33. 迭代第7次后為P4 
  34.  
  35. 越界枚舉測試結(jié)果 

真讓人吃驚,返回原始集合,越界之后就返回null了,但如果是MoveNext,不論有沒有加yield break, 越界迭代后還是返回***一個元素! 也許就是我們在第1節(jié)里提到的,迭代器只返回上一次的狀態(tài),因為無法后移,所以就重復(fù)返回,那為什么List集合就不會這樣呢?問題留給大家。

(感謝回答:越界枚舉到底是null還是***一個元素的問題,其實沒有明確規(guī)定,具體看.NET的實現(xiàn),在.NET Framework中,越界后依然是***一個元素)。

不過各位看官盡管放心,在foreach的標準枚舉過程下,枚舉是肯定能枚舉完的,這就說明了MoveNext和foreach兩種在實現(xiàn)上的不同,顯然foreach更安全。同時還注意,不能在yield過程中實現(xiàn)try-catch代碼塊,為什么呢?因為yield模式組合了來自不同位置的代碼和邏輯,怎么可能靠編譯給每個引用的代碼塊加上try-catch?這太復(fù)雜了。

枚舉的特性在處理大數(shù)據(jù)的時候很有幫助,就是因為它的狀態(tài)性,一個超大的文件,我只要每次讀一部分,就可以順次的讀取下去,直到文件結(jié)束,由于不需要實例化集合,內(nèi)存占用是很低的。對數(shù)據(jù)庫也是如此,每次讀取一部分,就能應(yīng)對很多難以應(yīng)付的情況。

3.在枚舉中修改枚舉器參數(shù)?

在枚舉過程中,集合是不能被修改的,比如在foreach循環(huán)中,如果插入或者刪除一個元素,肯定會報運行時異常。有經(jīng)驗的程序員告訴 你,此時用for循環(huán)。for和foreach的本質(zhì)區(qū)別是什么呢? 

在MoveNext中,我突然改變了枚舉的參數(shù),使得它的數(shù)據(jù)量變多或者變少了,又會發(fā)生什么?

  1. Console.WriteLine("不修改OldAge參數(shù)"); 
  2.             foreach (var olderPeople in peopleList.OlderPeoples) 
  3.             { 
  4.                 Console.WriteLine(olderPeople); 
  5.                
  6.             } 
  7.  
  8.             Console.WriteLine("修改了OldAge參數(shù)"); 
  9.             i = 0; 
  10.             foreach (var olderPeople in peopleList.OlderPeoples) 
  11.             { 
  12.                 Console.WriteLine(olderPeople); 
  13.                 i++; 
  14.                 if (i ==1) 
  15.                     peopleList.OldAge = 33;  //只枚舉一次后,修改OldAge 的值 
  16.             } 

測試結(jié)果是:

  1. 不修改OldAge參數(shù) 
  2. ID:2,NameP2,Age32 
  3. ID:3,NameP3,Age33 
  4. ID:4,NameP4,Age34 
  5.  
  6. 修改了OldAge參數(shù) 
  7. ID:2,NameP2,Age32 
  8. ID:4,NameP4,Age34 

可以看到,在枚舉過程中修改了控制枚舉的值,能動態(tài)改變枚舉的行為。上面是在一個yield結(jié)構(gòu)中改變變量的情況,我們再試試在迭代器和Lambda表達式的情況(代碼略), 得到結(jié)果是:

  1. 在迭代中修改變量值 
  2. ID:2,NameP2,Age32 
  3. ID:4,NameP4,Age34 
  4. 在Lambda表達式中修改變量值 
  5. ID:2,NameP2,Age32 
  6. ID:4,NameP4,Age34 

可以看出,外部修改變量能夠控制內(nèi)部的迭代過程,動態(tài)改變了“集合的元素”。 這是一個好事,因為它的行為確實是對的;也是壞事:在迭代過程中,修改了變量的值,上下文語境變化,可是如果還按之前的語境進行處理,顯然就會釀成大錯。 這里和閉包沒關(guān)系。

因此,如果一個枚舉需要在上下文會發(fā)生變化的情況下保持原有的行為,就需要手動保存變量的副本。

如果你把兩個集合A,B用Concat函數(shù)順次拼接起來,也就是A-B, 而且不實例化,那么在枚舉A的階段中,修改集合B的元素,會報錯么? 為什么?

比如如下的測試代碼:

  1. List<People> peoples=new List<People>(){new People(){Name = "PA"}}; 
  2.             Console.WriteLine("將一個虛擬枚舉A連接到集合B,并在枚舉A階段修改集合B的元素"); 
  3.             var e8 = peopleList.PeopleEnumerable1.Concat(peoples); 
  4.             i = 0
  5.             foreach (var people in e8) 
  6.             { 
  7.                 Console.WriteLine(people); 
  8.                 i++; 
  9.                 if (i == 1)    
  10.                   peoples.Add(new People(){Name = "PB"});  //此時還在枚舉PeopleEnumerable1階段 
  11.         } 

如果你想知道,可以自己做個試驗(在我附件里也有這個例子)。留給大家討論。

4. 更多LINQ的討論

你可以在yield中插入任何代碼,這就是延遲(Lazy)的表現(xiàn),只是需要執(zhí)行的時候才執(zhí)行。 我們不難想象Linq很多函數(shù)的實現(xiàn)方式,比較有意思的包括Concat,它將兩個集合連在了一起,就像下面這樣:

  1. public static IEnumerable<T> Concat<T>(this IEnumerable<T> source, IEnumerable<T> source2) 
  2.        { 
  3.            foreach (var r in source) 
  4.            { 
  5.                yield return r; 
  6.            } 
  7.            foreach (var r in source2) 
  8.            { 
  9.                yield return r; 
  10.            } 
  11.        } 

還有Select, Where都好實現(xiàn),就不討論了。

Skip怎么實現(xiàn)的呢?  它跳過了集合中的一部分元素,我猜是這樣的:

  1. public static IEnumerable<T> Skip<T>(this IEnumerable<T> source, int count) 
  2.        { 
  3.            int t = 0; 
  4.            foreach (var r in source) 
  5.            { 
  6.                t++; 
  7.                if(t<=count) 
  8.                    continue
  9.                yield return r; 
  10.            } 
  11.        } 

那么,被跳過的元素,到底被訪問過沒有?它的代碼被執(zhí)行了么?

  1. Console.WriteLine("Skip的元素是否會被訪問到?"); 
  2.  IEnumerable<People> e6 = peopleList.PeopleEnumerable1.Select(d => 
  3.        { 
  4.               Console.WriteLine(d); 
  5.               return d; 
  6.        }).Skip(3); 
  7.  Console.WriteLine("只枚舉,什么都不做:"); 
  8.  foreach (var  r in e6){}   
  9.  
  10.  Console.WriteLine("轉(zhuǎn)換為實體集合,再次枚舉"); 
  11.  IEnumerable<People> e7 = e6.ToList(); 
  12.  foreach (var r in e7){} 

測試結(jié)果如下:

  1. 只枚舉,什么都不做: 
  2. ID:0,NameP0,Age30 
  3. ID:1,NameP1,Age31 
  4. ID:2,NameP2,Age32 
  5. ID:3,NameP3,Age33 
  6. ID:4,NameP4,Age34 
  7. 轉(zhuǎn)換為實體集合,再次枚舉 
  8. ID:0,NameP0,Age30 
  9. ID:1,NameP1,Age31 
  10. ID:2,NameP2,Age32 
  11. ID:3,NameP3,Age33 
  12. ID:4,NameP4,Age34 

可以看出,Skip雖然是跳過,但還是會“訪問”元素的,因此會執(zhí)行額外的操作,比如lambda表達式,這不論是枚舉器還是實體集合都是如此。這個角度說,要優(yōu)化表達式,應(yīng)當盡可能在linq中早的Skip和Take,以減少額外的副作用。

但對于Linq to SQL的實現(xiàn)中,顯然Skip是做過額外優(yōu)化的。我們是否也能優(yōu)化Skip的實現(xiàn),使得上層盡可能提升海量數(shù)據(jù)下的Skip性能呢?

5. 有關(guān)IEnumerable枚舉的更多問題

(1) 枚舉過程如何暫停?有暫停這一說么? 如何取消?

(2) PLinq的實現(xiàn)原理是什么?它改變的到底是IEnumerable接口的哪種特性?是否產(chǎn)生了亂序枚舉?這種亂序枚舉到底是怎么實現(xiàn)?

(3) IEnumerable實現(xiàn)了鏈條結(jié)構(gòu),這是Linq的基礎(chǔ),但這個鏈條的本質(zhì)是什么?

(4) 因為IEnumerable代表了狀態(tài)和延遲,因此就不難理解很多異步操作的本質(zhì)就是IEnumerable。我有一次面試時候,問到了異步的實質(zhì),你說異步的實質(zhì)是什么?異步不是多線程!異步的精彩,本質(zhì)上是代碼的重新組合,因為長時間的異步操作就是狀態(tài)機。。。比如CCR庫。此處不準備展開說,因為暫時超過了作者的知識儲備,下次再說。

(5) 如果用C語言來實現(xiàn)同樣的枚舉器,同樣酷炫的Linq,不靠編譯器能實現(xiàn)么?先不提Lambda的梗,我們用函數(shù)指針。

(6) IEnumerable寫MapReduce? Linq for MapReduce?

(7) IEnumerable如何Sort? 實例化為一個集合再排序么?如果是一個超大的虛擬集合,如何優(yōu)化?

下一篇我們詳細討論這些內(nèi)容。附件是整個測試代碼,如果你覺得有幫助,請幫忙點推薦,謝謝.

完整測試代碼。

責任編輯:張偉 來源: 博客園
相關(guān)推薦

2012-11-23 10:57:44

Shell

2023-02-27 09:20:24

絕對定位CSS

2023-01-29 09:46:47

Dialog彈窗模態(tài)

2019-11-20 10:25:06

sudoLinux

2021-01-05 11:22:58

Python字符串代碼

2015-08-13 09:03:14

調(diào)試技巧

2020-01-29 19:40:36

Python美好,一直在身邊Line

2021-07-12 07:59:06

安全 HTML 屬性

2019-11-25 14:05:47

Python裝飾器數(shù)據(jù)

2021-12-17 00:10:00

ChromeDevtools功能

2020-03-05 11:10:18

Left join數(shù)據(jù)庫MySQL

2018-05-10 11:50:13

Docker容器冷知識

2024-03-04 00:00:00

Kubernetes技巧API

2016-09-05 13:14:11

2022-09-20 11:58:27

NpmNode.js

2010-07-29 09:18:31

Linux用戶

2015-05-14 15:59:33

DockerLinux容器管理工具

2010-07-21 12:37:11

Linux用戶

2020-05-09 08:48:21

JavaScript原生方法代碼

2011-02-14 16:11:44

點贊
收藏

51CTO技術(shù)棧公眾號

2014亚洲片线观看视频免费| 日韩中文欧美| 欧美视频在线观看免费| 日本a级片久久久| 一区二区www| 亚洲小说欧美另类婷婷| 国产视频精品一区二区三区| 亚洲黄色av网址| 午夜在线激情影院| 国产欧美一区二区在线观看| 91久久精品日日躁夜夜躁国产| 亚洲黄色免费在线观看| 国产成人免费精品| 亚洲午夜久久久久久久久电影院 | 日韩精品欧美专区| 97国产成人无码精品久久久| 激情文学一区| 国产一区二区三区在线观看网站 | 精品一区二区在线观看| 午夜精品福利在线观看| 一级片黄色录像| 欧美中文一区| 欧美一区二区三区白人| 粉嫩虎白女毛片人体| 欧美极品少妇videossex| 国产色产综合产在线视频| 97视频热人人精品| 亚洲一级在线播放| 久久精品女人天堂| 国自产精品手机在线观看视频| 免费毛片视频网站| 大陆精大陆国产国语精品| 91精品视频网| 在线观看av日韩| 最近高清中文在线字幕在线观看1| 国产精品国产三级国产普通话蜜臀| 国产精品二区三区四区| 国产美女www爽爽爽视频| 日本视频一区二区三区| 日本高清不卡在线| 国产又大又黄视频| 亚洲国产影院| 久久久久久中文| 国产探花在线播放| 在线看片不卡| 久国内精品在线| 无码人妻精品中文字幕 | 国内精品伊人久久久久av影院| 91豆花精品一区| 五月天婷婷网站| 伊人久久综合| 国内精品视频久久| 国产大片中文字幕| 精品动漫av| 韩国三级日本三级少妇99| 久久国产在线观看| 国产一区视频在线观看免费| 欧美日韩成人在线观看| 可以直接看的黄色网址| 亚洲国产精品日韩专区av有中文| 正在播放欧美视频| 精品熟妇无码av免费久久| 成人免费看片39| 中文综合在线观看| 少妇视频一区二区| 天天做天天爱综合| 久久精品国产亚洲7777| 黄视频网站免费看| 狠狠88综合久久久久综合网| 久久久久国产精品一区| 国产在线视频在线观看| 99热免费精品在线观看| 欧美一区二区三区……| 中文字幕在线日本| 久久99久久99| 99理论电影网| 天天干天天操av| 国产午夜精品久久久久久久| 伊人婷婷久久| 青草在线视频在线观看| 欧美日韩精品二区| 免费涩涩18网站入口| www.久久99| 日韩电影在线观看永久视频免费网站| 18禁一区二区三区| 羞羞色国产精品网站| 伊人激情综合网| 免费在线黄色片| 销魂美女一区二区三区视频在线| 国产97人人超碰caoprom| 国产一级精品毛片| 国产精品2024| 清纯唯美一区二区三区| 国产淫片在线观看| 欧美色xxxx| 欧美日韩精品区别| 久久影院资源站| 日韩中文字幕在线观看| 日韩经典在线观看| 久久精品国产精品亚洲精品| 国产视频一区二区不卡| yiren22亚洲综合伊人22| 亚洲综合av网| 国产精品久久久毛片| 国产精品视频3p| xxx欧美精品| 中文字幕国产在线观看| 国产精品66部| 一本久道久久综合| 中文在线资源| 日韩精品一区二区三区在线| 自拍偷拍你懂的| 亚洲久久成人| 91在线在线观看| 91精彩视频在线观看| 精品日韩美女的视频高清| 亚洲男人天堂2021| 精品国产1区| 国产91|九色| 亚洲精品久久久久久动漫器材一区| 久久精品一区二区三区不卡 | 欧美亚洲愉拍一区二区| 国产伦精品一区二区三区精品| 成人影视亚洲图片在线| 18性欧美xxxⅹ性满足| 国产成a人亚洲精v品无码| 欧美国产1区2区| 久草在在线视频| 欧美电影在线观看完整版| 九九热精品视频| 97国产精品久久久| 国产精品二区一区二区aⅴ污介绍| 国产色一区二区三区| 激情综合婷婷| 久久中文字幕国产| 在线观看中文字幕av| 国产亚洲精品精华液| 日韩av三级在线| 美女一区二区在线观看| 国内精品久久久久久| 亚洲高清精品视频| 一区二区三区在线视频观看58| 免费黄色一级网站| 国产一区2区| 国产成人精品一区二区| 激情小视频在线观看| 欧美午夜视频一区二区| 日韩 中文字幕| 免费久久99精品国产自在现线| 国产一区二区三区四区五区在线| 欧美极品少妇videossex| 精品久久久久久亚洲综合网 | 午夜视频一区二区| 久久久久亚洲av成人网人人软件| 91精品蜜臀一区二区三区在线| 国产精品久久久久久久久久| 8888四色奇米在线观看| 欧美日韩一区高清| chinese全程对白| 国产精品69毛片高清亚洲| wwwwww欧美| 日韩av黄色在线| 国产精品久久久久99| 麻豆视频免费在线观看| 欧美一个色资源| 国产精品theporn动漫| 99在线热播精品免费| 少妇无码av无码专区在线观看 | 欧美激情专区| 欧美成人精品一区二区男人小说| 亚洲精品一区中文| 亚洲天堂视频在线| 亚洲精品国产a| 国产伦精品一区三区精东| 性高湖久久久久久久久| 亚洲免费视频一区| 视频在线亚洲| 国产97在线视频| 在线中文字幕电影| 日韩高清欧美高清| 中文字幕一区二区三区免费看| 国产精品久久久久四虎| 无套白嫩进入乌克兰美女| 亚洲免费激情| 亚洲欧洲日夜超级视频| 久久久国产精品入口麻豆 | 精精国产xxxx视频在线播放| 精品呦交小u女在线| 亚洲天堂中文在线| 午夜在线电影亚洲一区| 91导航在线观看| 成人午夜激情在线| 91色国产在线| 亚洲手机在线| 欧美亚洲免费高清在线观看| 精品午夜视频| 国产精品狠色婷| 超黄网站在线观看| 色青青草原桃花久久综合| 深爱五月激情五月| 91精品国产欧美一区二区18| 永久免费无码av网站在线观看| 国产精品美女久久福利网站| 熟妇高潮一区二区| 国内久久精品视频| 成人性做爰aaa片免费看不忠| 亚洲情侣在线| 日韩国产精品一区二区| 国内精品偷拍| 亚洲a中文字幕| 欧美极品免费| 性欧美亚洲xxxx乳在线观看| 毛片在线看网站| 亚洲色图综合久久| 天天干免费视频| 欧美α欧美αv大片| 一级黄色片免费| 色悠悠亚洲一区二区| 久久精品视频日本| 亚洲色图制服诱惑| 少妇太紧太爽又黄又硬又爽小说| 9色porny自拍视频一区二区| 五月天国产视频| 麻豆国产精品777777在线| 成人在线激情网| 噜噜噜91成人网| 欧美日韩精品在线一区二区| 国产一区亚洲| 特色特色大片在线| 午夜精品一区二区三区国产| 日本一区二区三区免费观看| 网红女主播少妇精品视频| 国产伦精品一区二区三| 亚洲一二av| 成人资源av| 91精品短视频| 国产精品一区二| 99久热这里只有精品视频免费观看| 91精品久久久久久久久久久久久久| 不卡一二三区| 日韩av电影在线播放| av电影一区| 国产成人精品a视频一区www| 国产日韩电影| 国产99久久精品一区二区永久免费 | 免费成人蒂法网站| 东方欧美亚洲色图在线| 亚洲综合伊人久久| 激情综合网av| aaaaaaaa毛片| 粉嫩高潮美女一区二区三区| 国产成人av免费观看| 国产精品资源在线| 催眠调教后宫乱淫校园| 99久久99久久精品国产片果冻| 麻豆精品国产传媒| 成人午夜免费视频| 国产精品无码久久久久久| 成人午夜av在线| 国产亚洲色婷婷久久99精品91| 成人午夜激情影院| 中国黄色a级片| 久久精品一区八戒影视| 天天操天天干天天操天天干| 国产精品你懂的| 美女视频黄免费| 精品福利一区二区| 在线免费观看av网址| 欧美日韩在线精品一区二区三区激情 | 夜夜嗨aⅴ一区二区三区| 欧美久久一二三四区| 精品欧美一区二区精品少妇| 亚洲成人亚洲激情| 国产精品影院在线| 免费成人高清视频| 天堂av中文在线观看| 国产精品高潮呻吟久久av无限| 国产精品伦理| 国产综合福利在线| 国产精品videossex| 欧美精品一区在线发布| 99精品美女| 国产在线播放观看| 日本色综合中文字幕| 三级黄色片免费观看| 99在线精品一区二区三区| 国产一区二区三区四区在线| 亚洲精品欧美在线| 日日噜噜噜噜人人爽亚洲精品| 欧美亚洲综合另类| 亚洲精品久久久狠狠狠爱| 亚洲视频专区在线| 性欧美videos高清hd4k| 国产91亚洲精品| 日韩中文字幕在线一区| 欧美性色黄大片人与善| 国产综合自拍| 538任你躁在线精品免费| 成人免费视频一区二区| 精品人妻中文无码av在线| 亚洲国产综合91精品麻豆| 中文字幕人妻互换av久久| 精品国产伦一区二区三区观看体验 | 欧美日韩视频| 久久综合久久色| 国产电影一区二区三区| x88av在线| 午夜欧美大尺度福利影院在线看| 久久影视中文字幕| 亚洲成年人在线| www.在线视频| 国产欧美一区二区三区在线看| 久久亚洲黄色| 久久www视频| 久久精品国产免费| 波多野吉衣中文字幕| 亚洲一二三四久久| 国产老妇伦国产熟女老妇视频| 日韩国产一区三区| 欧美v亚洲v| 91影视免费在线观看| 成人精品视频| 日日碰狠狠躁久久躁婷婷| aa级大片欧美| 国产精品第九页| 日韩欧美一二三四区| 毛片免费不卡| 91精品久久久久久久久久另类| 国产精品三级| 精品中文字幕av| 99精品视频在线播放观看| 久久黄色免费视频| 日韩一卡二卡三卡四卡| 69久久夜色| 国产精品人成电影| 国语产色综合| 成人黄色一区二区| 日本一区二区三区在线观看| caoporn国产| 亚洲欧洲午夜一线一品| 在线天堂资源| 免费精品视频一区| 久久精品国产清高在天天线| 特级西西人体wwwww| 黑人巨大精品欧美一区二区三区 | 精品国产91乱码一区二区三区| 久操免费在线| 3d动漫精品啪啪一区二区三区免费| 人人狠狠综合久久亚洲婷| 99视频精品免费| 国产精品福利电影一区二区三区四区| 日韩精品视频免费播放| 亚洲激情第一页| 在线免费三级电影网站| 欧洲亚洲一区二区三区四区五区| 亚洲免费网站| 国产又大又粗又爽的毛片| 欧美日韩国产一二三| 久草中文在线| 国产精品免费一区二区三区四区| 欧美在线日韩| 午夜视频在线观看国产| 黑人巨大精品欧美一区免费视频| av女名字大全列表| 国产成人一区二区三区电影| 色喇叭免费久久综合| 韩国三级hd中文字幕有哪些| 亚洲地区一二三色| 六十路在线观看| 国产欧美一区二区三区久久人妖| 久久久五月天| 亚洲婷婷在线观看| 一本大道久久精品懂色aⅴ| 福利片在线看| 亚洲影院高清在线| 日韩亚洲在线| 大胸美女被爆操| 精品国产伦一区二区三区观看方式| 福利在线免费视频| 色999日韩自偷自拍美女| 国产伦精品一区二区三区视频青涩 | 一级片视频免费观看| 日韩毛片高清在线播放| 午夜老司机福利| 97精品国产97久久久久久免费 | 亚洲精品美女久久| jizzyou欧美16| 国产1区2区3区中文字幕| 91老司机福利 在线| 国产精品无码久久av| 国产91精品青草社区| 天天射—综合中文网| 久久久久久久久免费看无码| 在线不卡的av| 伊人久久精品一区二区三区| 国产成年人在线观看| 97久久精品人人做人人爽50路| 亚洲 小说区 图片区| 久久欧美在线电影| 91日韩视频|