韩剧1988免费观看全集_久久影视三级福利片_亚洲视频在线观看免费_在线观看欧美日韩_国产亚洲激情在线_亚洲精品美女久久久_欧美国产日韩一区二区在线观看_91在线观看免费高清完整版在线观看_日韩av免费看_国产又爽又黄的激情精品视频_琪琪亚洲精品午夜在线_欧美性猛xxx_不卡毛片在线看_国产亚洲日本欧美韩国_91国内在线视频_精品国产福利视频

當(dāng)前位置:蘿卜系統(tǒng)下載站 > 技術(shù)開發(fā)教程 > 詳細(xì)頁面

如何在C#中運(yùn)用 Win32與其他庫

如何在C#中運(yùn)用 Win32與其他庫

更新時間:2022-06-07 文章作者:未知 信息來源:網(wǎng)絡(luò) 閱讀次數(shù):

C# 用戶經(jīng)常提出兩個問題:“我為什么要另外編寫代碼來使用內(nèi)置于 Windows 中的功能?在框架中為什么沒有相應(yīng)的內(nèi)容可以為我完成這一任務(wù)?”當(dāng)框架小組構(gòu)建他們的 .NET 部分時,他們評估了為使 .NET 程序員可以使用 Win32 而需要完成的工作,結(jié)果發(fā)現(xiàn) Win32 API 集非常龐大。他們沒有足夠的資源為所有 Win32 API 編寫托管接口、加以測試并編寫文檔,因此只能優(yōu)先處理最重要的部分。許多常用操作都有托管接口,但是還有許多完整的 Win32 部分沒有托管接口。

  平臺調(diào)用 (P/Invoke) 是完成這一任務(wù)的最常用方法。要使用 P/Invoke,您可以編寫一個描述如何調(diào)用函數(shù)的原型,然后運(yùn)行時將使用此信息進(jìn)行調(diào)用。另一種方法是使用 Managed Extensions to C++ 來包裝函數(shù),這部分內(nèi)容將在以后的專欄中介紹。

  要理解如何完成這一任務(wù),最好的辦法是通過示例。在某些示例中,我只給出了部分代碼;完整的代碼可以通過下載獲得。

  簡單示例

  在第一個示例中,我們將調(diào)用 Beep() API 來發(fā)出聲音。首先,我需要為 Beep() 編寫適當(dāng)?shù)亩x。查看 MSDN 中的定義,我發(fā)現(xiàn)它具有以下原型:

  BOOL Beep(
 DWORD dwFreq,   // 聲音頻率
 DWORD dwDuration  // 聲音持續(xù)時間
);


  要用 C# 來編寫這一原型,需要將 Win32 類型轉(zhuǎn)換成相應(yīng)的 C# 類型。由于 DWORD 是 4 字節(jié)的整數(shù),因此我們可以使用 int 或 uint 作為 C# 對應(yīng)類型。由于 int 是 CLS 兼容類型(可以用于所有 .NET 語言),以此比 uint 更常用,并且在多數(shù)情況下,它們之間的區(qū)別并不重要。bool 類型與 BOOL 對應(yīng),F(xiàn)在我們可以用 C# 編寫以下原型:

  public static extern bool Beep(int frequency, int duration);

  這是相當(dāng)標(biāo)準(zhǔn)的定義,只不過我們使用了 extern 來指明該函數(shù)的實(shí)際代碼在別處。此原型將告訴運(yùn)行時如何調(diào)用函數(shù);現(xiàn)在我們需要告訴它在何處找到該函數(shù)。

  我們需要回顧一下 MSDN 中的代碼。在參考信息中,我們發(fā)現(xiàn) Beep() 是在 kernel32.lib 中定義的。這意味著運(yùn)行時代碼包含在 kernel32.dll 中。我們在原型中添加 DllImport 屬性將這一信息告訴運(yùn)行時:

  [DllImport("kernel32.dll")]

  這就是我們要做的全部工作。下面是一個完整的示例,它生成的隨機(jī)聲音在二十世紀(jì)六十年代的科幻電影中很常見。

 

using System;
using System.Runtime.InteropServices;

namespace Beep
{
class Class1
 {
   [DllImport("kernel32.dll")]
   public static extern bool Beep(int frequency, int duration);

   static void Main(string[] args)
   {
     Random random = new Random();

     for (int i = 0; i < 10000; i++)
     {
      Beep(random.Next(10000), 100);
}
   }
 }
}


  它的聲響足以刺激任何聽者!由于 DllImport 允許您調(diào)用 Win32 中的任何代碼,因此就有可能調(diào)用惡意代碼。所以您必須是完全受信任的用戶,運(yùn)行時才能進(jìn)行 P/Invoke 調(diào)用。

  枚舉和常量

  Beep() 可用于發(fā)出任意聲音,但有時我們希望發(fā)出特定類型的聲音,因此我們改用 MessageBeep()。MSDN 給出了以下原型:

 

BOOL MessageBeep(
 UINT uType // 聲音類型
);


  這看起來很簡單,但是從注釋中可以發(fā)現(xiàn)兩個有趣的事實(shí)。

  首先,uType 參數(shù)實(shí)際上接受一組預(yù)先定義的常量。

  其次,可能的參數(shù)值包括 -1,這意味著盡管它被定義為 uint 類型,但 int 會更加適合。

  對于 uType 參數(shù),使用 enum 類型是合乎情理的。MSDN 列出了已命名的常量,但沒有就具體值給出任何提示。由于這一點(diǎn),我們需要查看實(shí)際的 API。

  如果您安裝了 Visual Studio? 和 C++,則 Platform SDK 位于 Program FilesMicrosoft Visual Studio .NETVc7PlatformSDKInclude 下。

  為查找這些常量,我在該目錄中執(zhí)行了一個 findstr。

  findstr "MB_ICONHAND" *.h

  它確定了常量位于 winuser.h 中,然后我使用這些常量來創(chuàng)建我的 enum 和原型:

 

public enum BeepType
{
  SimpleBeep = -1,
  IconAsterisk = 0x00000040,
  IconExclamation = 0x00000030,
  IconHand = 0x00000010,
  IconQuestion = 0x00000020,
  Ok = 0x00000000,
}

[DllImport("user32.dll")]
public static extern bool MessageBeep(BeepType beepType);


  現(xiàn)在我可以用下面的語句來調(diào)用它: MessageBeep(BeepType.IconQuestion);

處理結(jié)構(gòu)

  有時我需要確定我筆記本的電池狀況。Win32 為此提供了電源管理函數(shù)。

  搜索 MSDN 可以找到 GetSystemPowerStatus() 函數(shù)。

 

BOOL GetSystemPowerStatus(
 LPSYSTEM_POWER_STATUS lpSystemPowerStatus
);


  此函數(shù)包含指向某個結(jié)構(gòu)的指針,我們尚未對此進(jìn)行過處理。要處理結(jié)構(gòu),我們需要用 C# 定義結(jié)構(gòu)。我們從非托管的定義開始:

 

typedef struct _SYSTEM_POWER_STATUS {
BYTE  ACLineStatus;
BYTE  BatteryFlag;
BYTE  BatteryLifePercent;
BYTE  Reserved1;
DWORD BatteryLifeTime;
DWORD BatteryFullLifeTime;
} SYSTEM_POWER_STATUS, *LPSYSTEM_POWER_STATUS;


  然后,通過用 C# 類型代替 C 類型來得到 C# 版本。

 

struct SystemPowerStatus
{
  byte ACLineStatus;
  byte batteryFlag;
  byte batteryLifePercent;
  byte reserved1;
  int batteryLifeTime;
  int batteryFullLifeTime;
}


  這樣,就可以方便地編寫出 C# 原型:

 

[DllImport("kernel32.dll")]
public static extern bool GetSystemPowerStatus(
  ref SystemPowerStatus systemPowerStatus);


  在此原型中,我們用“ref”指明將傳遞結(jié)構(gòu)指針而不是結(jié)構(gòu)值。這是處理通過指針傳遞的結(jié)構(gòu)的一般方法。

  此函數(shù)運(yùn)行良好,但是最好將 ACLineStatus 和 batteryFlag 字段定義為 enum:

 

  enum ACLineStatus: byte
  {
   Offline = 0,
   Online = 1,
   Unknown = 255,
  }

  enum BatteryFlag: byte
  {
   High = 1,
   Low = 2,
   Critical = 4,
   Charging = 8,
   NoSystemBattery = 128,
   Unknown = 255,
  }


  請注意,由于結(jié)構(gòu)的字段是一些字節(jié),因此我們使用 byte 作為該 enum 的基本類型。

  字符串

  雖然只有一種 .NET 字符串類型,但這種字符串類型在非托管應(yīng)用中卻有幾項獨(dú)特之處?梢允褂镁哂袃(nèi)嵌字符數(shù)組的字符指針和結(jié)構(gòu),其中每個數(shù)組都需要正確的封送處理。

  在 Win32 中還有兩種不同的字符串表示:

  ANSI
  Unicode

  最初的 Windows 使用單字節(jié)字符,這樣可以節(jié)省存儲空間,但在處理很多語言時都需要復(fù)雜的多字節(jié)編碼。Windows NT? 出現(xiàn)后,它使用雙字節(jié)的 Unicode 編碼。為解決這一差別,Win32 API 采用了非常聰明的做法。它定義了 TCHAR 類型,該類型在 Win9x 平臺上是單字節(jié)字符,在 WinNT 平臺上是雙字節(jié) Unicode 字符。對于每個接受字符串或結(jié)構(gòu)(其中包含字符數(shù)據(jù))的函數(shù),Win32 API 均定義了該結(jié)構(gòu)的兩種版本,用 A 后綴指明 Ansi 編碼,用 W 指明 wide 編碼(即 Unicode)。如果您將 C++ 程序編譯為單字節(jié),會獲得 A 變體,如果編譯為 Unicode,則獲得 W 變體。Win9x 平臺包含 Ansi 版本,而 WinNT 平臺則包含 W 版本。

  由于 P/Invoke 的設(shè)計者不想讓您為所在的平臺操心,因此他們提供了內(nèi)置的支持來自動使用 A 或 W 版本。如果您調(diào)用的函數(shù)不存在,互操作層將為您查找并使用 A 或 W 版本。

  通過示例能夠很好地說明字符串支持的一些精妙之處。

簡單字符串

  下面是一個接受字符串參數(shù)的函數(shù)的簡單示例:

 

BOOL GetDiskFreeSpace(
LPCTSTR lpRootPathName,     // 根路徑
LPDWORD lpSectorsPerCluster,  // 每個簇的扇區(qū)數(shù)
LPDWORD lpBytesPerSector,    // 每個扇區(qū)的字節(jié)數(shù)
LPDWORD lpNumberOfFreeClusters, // 可用的扇區(qū)數(shù)
LPDWORD lpTotalNumberOfClusters // 扇區(qū)總數(shù)
);


  根路徑定義為 LPCTSTR。這是獨(dú)立于平臺的字符串指針。

  由于不存在名為 GetDiskFreeSpace() 的函數(shù),封送拆收器將自動查找“A”或“W”變體,并調(diào)用相應(yīng)的函數(shù)。我們使用一個屬性來告訴封送拆收器,API 所要求的字符串類型。

  以下是該函數(shù)的完整定義,就象我開始定義的那樣:

 

[DllImport("kernel32.dll")]
static extern bool GetDiskFreeSpace(
 [MarshalAs(UnmanagedType.LPTStr)]
 string rootPathName,
  ref int sectorsPerCluster,
  ref int bytesPerSector,
  ref int numberOfFreeClusters,
  ref int totalNumberOfClusters);


  不幸的是,當(dāng)我試圖運(yùn)行時,該函數(shù)不能執(zhí)行。問題在于,無論我們在哪個平臺上,封送拆收器在默認(rèn)情況下都試圖查找 API 的 Ansi 版本,由于 LPTStr 意味著在 Windows NT 平臺上會使用 Unicode 字符串,因此試圖用 Unicode 字符串來調(diào)用 Ansi 函數(shù)就會失敗。

有兩種方法可以解決這個問題:一種簡單的方法是刪除 MarshalAs 屬性。如果這樣做,將始終調(diào)用該函數(shù)的 A 版本,如果在您所涉及的所有平臺上都有這種版本,這是個很好的方法。但是,這會降低代碼的執(zhí)行速度,因為封送拆收器要將 .NET 字符串從 Unicode 轉(zhuǎn)換為多字節(jié),然后調(diào)用函數(shù)的 A 版本(將字符串轉(zhuǎn)換回 Unicode),最后調(diào)用函數(shù)的 W 版本。

  要避免出現(xiàn)這種情況,您需要告訴封送拆收器,要它在 Win9x 平臺上時查找 A 版本,而在 NT 平臺上時查找 W 版本。要實(shí)現(xiàn)這一目的,可以將 CharSet 設(shè)置為 DllImport 屬性的一部分:

  [DllImport("kernel32.dll", CharSet = CharSet.Auto)]

  在我的非正式計時測試中,我發(fā)現(xiàn)這一做法比前一種方法快了大約百分之五。

  對于大多數(shù) Win32 API,都可以對字符串類型設(shè)置 CharSet 屬性并使用 LPTStr。但是,還有一些不采用 A/W 機(jī)制的函數(shù),對于這些函數(shù)必須采取不同的方法。

字符串緩沖區(qū)

  .NET 中的字符串類型是不可改變的類型,這意味著它的值將永遠(yuǎn)保持不變。對于要將字符串值復(fù)制到字符串緩沖區(qū)的函數(shù),字符串將無效。這樣做至少會破壞由封送拆收器在轉(zhuǎn)換字符串時創(chuàng)建的臨時緩沖區(qū);嚴(yán)重時會破壞托管堆,而這通常會導(dǎo)致錯誤的發(fā)生。無論哪種情況都不可能獲得正確的返回值。

  要解決此問題,我們需要使用其他類型。StringBuilder 類型就是被設(shè)計為用作緩沖區(qū)的,我們將使用它來代替字符串。下面是一個示例:

 

[DllImport("kernel32.dll", CharSet = CharSet.Auto)]
public static extern int GetShortPathName(
  [MarshalAs(UnmanagedType.LPTStr)]
  string path,
  [MarshalAs(UnmanagedType.LPTStr)]
  StringBuilder shortPath,
  int shortPathLength);


  使用此函數(shù)很簡單:

 

StringBuilder shortPath = new StringBuilder(80);
int result = GetShortPathName(
@"d: est.jpg", shortPath, shortPath.Capacity);
string s = shortPath.ToString();


  請注意,StringBuilder 的 Capacity 傳遞的是緩沖區(qū)大小。

  具有內(nèi)嵌字符數(shù)組的結(jié)構(gòu)

  某些函數(shù)接受具有內(nèi)嵌字符數(shù)組的結(jié)構(gòu)。例如,GetTimeZoneInformation() 函數(shù)接受指向以下結(jié)構(gòu)的指針:

 

typedef struct _TIME_ZONE_INFORMATION {
  LONG    Bias;
  WCHAR   StandardName[ 32 ];
  SYSTEMTIME StandardDate;
  LONG    StandardBias;
  WCHAR   DaylightName[ 32 ];
  SYSTEMTIME DaylightDate;
  LONG    DaylightBias;
} TIME_ZONE_INFORMATION, *PTIME_ZONE_INFORMATION;


  在 C# 中使用它需要有兩種結(jié)構(gòu)。一種是 SYSTEMTIME,它的設(shè)置很簡單:

 

  struct SystemTime
  {
   public short wYear;
   public short wMonth;
   public short wDayOfWeek;
   public short wDay;
   public short wHour;
   public short wMinute;
   public short wSecond;
   public short wMilliseconds;
  }


  這里沒有什么特別之處;另一種是 TimeZoneInformation,它的定義要復(fù)雜一些:

 

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
struct TimeZoneInformation
{
  public int bias;
  [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
  public string standardName;
  SystemTime standardDate;
  public int standardBias;
  [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]
  public string daylightName;
  SystemTime daylightDate;
  public int daylightBias;
}


  此定義有兩個重要的細(xì)節(jié)。第一個是 MarshalAs 屬性:

  [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 32)]

  查看 ByValTStr 的文檔,我們發(fā)現(xiàn)該屬性用于內(nèi)嵌的字符數(shù)組;另一個是 SizeConst,它用于設(shè)置數(shù)組的大小。

  我在第一次編寫這段代碼時,遇到了執(zhí)行引擎錯誤。通常這意味著部分互操作覆蓋了某些內(nèi)存,表明結(jié)構(gòu)的大小存在錯誤。我使用 Marshal.SizeOf() 來獲取所使用的封送拆收器的大小,結(jié)果是 108 字節(jié)。我進(jìn)一步進(jìn)行了調(diào)查,很快回憶起用于互操作的默認(rèn)字符類型是 Ansi 或單字節(jié)。而函數(shù)定義中的字符類型為 WCHAR,是雙字節(jié),因此導(dǎo)致了這一問題。

  我通過添加 StructLayout 屬性進(jìn)行了更正。結(jié)構(gòu)在默認(rèn)情況下按順序布局,這意味著所有字段都將以它們列出的順序排列。CharSet 的值被設(shè)置為 Unicode,以便始終使用正確的字符類型。

  經(jīng)過這樣處理后,該函數(shù)一切正常。您可能想知道我為什么不在此函數(shù)中使用 CharSet.Auto。這是因為,它也沒有 A 和 W 變體,而始終使用 Unicode 字符串,因此我采用了上述方法編碼。

具有回調(diào)的函數(shù)

  當(dāng) Win32 函數(shù)需要返回多項數(shù)據(jù)時,通常都是通過回調(diào)機(jī)制來實(shí)現(xiàn)的。開發(fā)人員將函數(shù)指針傳遞給函數(shù),然后針對每一項調(diào)用開發(fā)人員的函數(shù)。

  在 C# 中沒有函數(shù)指針,而是使用“委托”,在調(diào)用 Win32 函數(shù)時使用委托來代替函數(shù)指針。

  EnumDesktops() 函數(shù)就是這類函數(shù)的一個示例:

 

BOOL EnumDesktops(
 HWINSTA hwinsta,       // 窗口實(shí)例的句柄
 DESKTOPENUMPROC lpEnumFunc, // 回調(diào)函數(shù)
 LPARAM lParam        // 用于回調(diào)函數(shù)的值
);


  HWINSTA 類型由 IntPtr 代替,而 LPARAM 由 int 代替。DESKTOPENUMPROC 所需的工作要多一些。下面是 MSDN 中的定義:

 

BOOL CALLBACK EnumDesktopProc(
 LPTSTR lpszDesktop, // 桌面名稱
 LPARAM lParam    // 用戶定義的值
);


  我們可以將它轉(zhuǎn)換為以下委托:

 

delegate bool EnumDesktopProc(
 [MarshalAs(UnmanagedType.LPTStr)]
  string desktopName,
  int lParam);


  完成該定義后,我們可以為 EnumDesktops() 編寫以下定義:

 

[DllImport("user32.dll", CharSet = CharSet.Auto)]
static extern bool EnumDesktops(
  IntPtr windowStation,
  EnumDesktopProc callback,
  int lParam);


  這樣該函數(shù)就可以正常運(yùn)行了。

  在互操作中使用委托時有個很重要的技巧:封送拆收器創(chuàng)建了指向委托的函數(shù)指針,該函數(shù)指針被傳遞給非托管函數(shù)。但是,封送拆收器無法確定非托管函數(shù)要使用函數(shù)指針做些什么,因此它假定函數(shù)指針只需在調(diào)用該函數(shù)時有效即可。

  結(jié)果是如果您調(diào)用諸如 SetConsoleCtrlHandler() 這樣的函數(shù),其中的函數(shù)指針將被保存以便將來使用,您就需要確保在您的代碼中引用委托。如果不這樣做,函數(shù)可能表面上能執(zhí)行,但在將來的內(nèi)存回收處理中會刪除委托,并且會出現(xiàn)錯誤。

  其他高級函數(shù)

  迄今為止我列出的示例都比較簡單,但是還有很多更復(fù)雜的 Win32 函數(shù)。下面是一個示例:

 

DWORD SetEntriesInAcl(
 ULONG cCountOfExplicitEntries,      // 項數(shù)
 PEXPLICIT_ACCESS pListOfExplicitEntries, // 緩沖區(qū)
 PACL OldAcl,               // 原始 ACL
 PACL *NewAcl               // 新 ACL
);


  前兩個參數(shù)的處理比較簡單:ulong 很簡單,并且可以使用 UnmanagedType.LPArray 來封送緩沖區(qū)。

  但第三和第四個參數(shù)有一些問題。問題在于定義 ACL 的方式。ACL 結(jié)構(gòu)僅定義了 ACL 標(biāo)頭,而緩沖區(qū)的其余部分由 ACE 組成。ACE 可以具有多種不同類型,并且這些不同類型的 ACE 的長度也不同。

  如果您愿意為所有緩沖區(qū)分配空間,并且愿意使用不太安全的代碼,則可以用 C# 進(jìn)行處理。但工作量很大,并且程序非常難調(diào)試。而使用 C++ 處理此 API 就容易得多。

  屬性的其他選項

  DLLImport 和 StructLayout 屬性具有一些非常有用的選項,有助于 P/Invoke 的使用。下面列出了所有這些選項:

  DLLImport

  CallingConvention

  您可以用它來告訴封送拆收器,函數(shù)使用了哪些調(diào)用約定。您可以將它設(shè)置為您的函數(shù)的調(diào)用約定。通常,如果此設(shè)置錯誤,代碼將不能執(zhí)行。但是,如果您的函數(shù)是 Cdecl 函數(shù),并且使用 StdCall(默認(rèn))來調(diào)用該函數(shù),那么函數(shù)能夠執(zhí)行,但函數(shù)參數(shù)不會從堆棧中刪除,這會導(dǎo)致堆棧被填滿。

  CharSet

  控制調(diào)用 A 變體還是調(diào)用 W 變體。

  EntryPoint

  此屬性用于設(shè)置封送拆收器在 DLL 中查找的名稱。設(shè)置此屬性后,您可以將 C# 函數(shù)重新命名為任何名稱。

  ExactSpelling

  將此屬性設(shè)置為 true,封送拆收器將關(guān)閉 A 和 W 的查找特性。

  PreserveSig

  COM 互操作使得具有最終輸出參數(shù)的函數(shù)看起來是由它返回的該值。此屬性用于關(guān)閉這一特性。

  SetLastError

  確保調(diào)用 Win32 API SetLastError(),以便您找出發(fā)生的錯誤。

  StructLayout

  LayoutKind

  結(jié)構(gòu)在默認(rèn)情況下按順序布局,并且在多數(shù)情況下都適用。如果需要完全控制結(jié)構(gòu)成員所放置的位置,可以使用 LayoutKind.Explicit,然后為每個結(jié)構(gòu)成員添加 FieldOffset 屬性。當(dāng)您需要創(chuàng)建 union 時,通常需要這樣做。

  CharSet

  控制 ByValTStr 成員的默認(rèn)字符類型。

  Pack

  設(shè)置結(jié)構(gòu)的壓縮大小。它控制結(jié)構(gòu)的排列方式。如果 C 結(jié)構(gòu)采用了其他壓縮方式,您可能需要設(shè)置此屬性。

  Size

  設(shè)置結(jié)構(gòu)大小。不常用;但是如果需要在結(jié)構(gòu)末尾分配額外的空間,則可能會用到此屬性。

  從不同位置加載

  您無法指定希望 DLLImport 在運(yùn)行時從何處查找文件,但是可以利用一個技巧來達(dá)到這一目的。

  DllImport 調(diào)用 LoadLibrary() 來完成它的工作。如果進(jìn)程中已經(jīng)加載了特定的 DLL,那么即使指定的加載路徑不同,LoadLibrary() 也會成功。

  這意味著如果直接調(diào)用 LoadLibrary(),您就可以從任何位置加載 DLL,然后 DllImport LoadLibrary() 將使用該 DLL。

  由于這種行為,我們可以提前調(diào)用 LoadLibrary(),從而將您的調(diào)用指向其他 DLL。如果您在編寫庫,可以通過調(diào)用 GetModuleHandle() 來防止出現(xiàn)這種情況,以確保在首次調(diào)用 P/Invoke 之前沒有加載該庫。

  P/Invoke 疑難解答

  如果您的 P/Invoke 調(diào)用失敗,通常是因為某些類型的定義不正確。以下是幾個常見問題:

  1.long != long。在 C++ 中,long 是 4 字節(jié)的整數(shù),但在 C# 中,它是 8 字節(jié)的整數(shù)。

  2.字符串類型設(shè)

溫馨提示:喜歡本站的話,請收藏一下本站!

本類教程下載

系統(tǒng)下載排行

網(wǎng)站地圖xml | 網(wǎng)站地圖html
领导边摸边吃奶边做爽在线观看| 色综合久久悠悠| 国产欧美久久久久久久久| 欧美激情精品在线| 性欧美freesex顶级少妇| 欧美jizz| 久久久久国产一区二区三区| 一色桃子久久精品亚洲| 中国女人内谢69视频| 麻豆成人入口| 亚洲www色| 国产青草视频在线观看| 精品视频www| 国产丝袜精品视频| av色综合网| 亚洲国产高清福利视频| 天堂资源最新在线| 美女视频网站在线观看| 国产在线一区二区三区四区| 久久国产精品成人免费观看的软件| 国产精品久久久久久久免费大片| 欧美亚洲大片| 日本一本不卡| 又黄又湿又爽又免费又色| 国产粉嫩在线观看| aa国产成人| 在线精品亚洲一区二区不卡| 久久精品国产亚洲av麻豆色欲| 国产日韩视频在线播放| 成人欧美一区二区三区在线观看| 特级西西444www高清大视频| 99在线精品视频在线观看| 亚洲色婷婷久久精品av蜜桃| 人人澡人人透人人爽| 日本在线视频站| 色综合久久中文综合久久牛| 91精品久久久久久蜜臀| 欧美中日韩一区二区三区| 91美女福利视频高清| 亚洲v日韩v综合v精品v| 夜夜骑日日操| 高清一区二区| 国产v综合v亚洲欧美久久| aaaa一级片| 日韩影视一区二区三区| 在线亚洲不卡| 视频一区二区中文字幕| 欧美黄色网视频| 四虎国产精品成人免费入口| 邪恶网站在线观看| 日韩欧美在线精品| 祥仔av免费一区二区三区四区| 韩国成人一区| 黄色片免费在线观看视频| 欧美性色黄大片手机版| 怡红院av在线| 91精品在线观看入口| 亚洲五月激情网| 国产青草视频在线观看视频| 成人激情视频在线| 欧美色黄视频| 91九色porn在线资源| 国产人妻精品午夜福利免费| 国产欧美日韩精品一区| 久久日韩粉嫩一区二区三区| 爱爱精品视频| 在线免费观看亚洲视频| 成人交换视频| 欧美gay囗交囗交| 一区二区三区鲁丝不卡| 成人免费一区二区三区在线观看| 全部av―极品视觉盛宴亚洲| 国产精品女人网站| 国产精品伦一区二区三级视频| 5858p先锋影音资源网| www.成人网.com| 亚洲精品ww久久久久久p站| 亚洲自拍另类综合| 美女黄a一级视频| 国产三级在线观看完整版| 激情综合自拍| 1区2区3区在线视频| 久久久久久久久蜜桃| 暖暖在线中文免费日本| 一本大道东京热无码aⅴ| 91在线无精精品白丝| 国产精品成人久久久久| 久久天堂av综合合色| 亚洲欧洲av另类| 欧美特级黄色片| 一区二区三区欧美在线| 国产精品久久一区二区三区不卡| 中国成人一区| 色综合久久久久综合体| 私人影视中文字幕| 丰满少妇被猛烈进入一区二区| 超碰97在线免费观看| 91精品国产91| 亚洲男人天堂久久| 亚洲人成网站在线播放2019| 久久午夜老司机| 日韩一级大片在线| 手机在线国产视频| 91国在线产| 国产人妖ts一区二区| 亚洲精品国产a| 在线观看免费高清视频97| 欧美性生活久久| 日本乱人伦一区| 日本网站在线观看一区二区三区| 人与牲动交xxxxbbb| 久久91精品国产91久久小草| 久久美女免费视频| 久久五月激情| 国产视频一区二区| 亚洲中文字幕无码av| 午夜免费福利网站| 九九热这里只有| 清纯唯美亚洲色图| 好吊操这里只有精品| 亚洲午夜视频在线| 免费成人高清在线视频theav| 激情综合色播激情啊| 韩国精品在线观看| аⅴ天堂中文在线网| 国产视频123区| 免费a级人成a大片在线观看| 欧美日韩精品一区二区视频| 午夜免费久久看| 日韩 欧美 高清| 国产chinesehd精品露脸| 97久久精品人人做人人爽| 亚洲精品在线免费播放| 自慰无码一区二区三区| 久热精品视频在线播放| 狠狠97人人婷婷五月| 亚洲a∨一区二区三区| 日韩一区三区| 日韩成人在线视频| 亚洲韩国一区二区三区| www欧美成人18+| 欧洲成人午夜精品无码区久久| 久久久久综合一区二区三区| 国产视频xxxx| 丝袜美腿一区| 鲁大师成人一区二区三区| 在线播放日韩导航| 天天操精品视频| 中文字幕中文字幕| 亚洲精品精品亚洲| 一个人在线视频免费观看www| 亚洲三级性片| 欧美成人小视频| 亚洲视频网站在线观看| 欧美伦理免费在线| 美女张开让男人捅| 777色狠狠一区二区三区| 高清国产一区二区| 欧美成人免费全部网站| 国产乱码精品一区二三赶尸艳谈| 日韩一区二区久久| 亚洲精品视频免费| 中文字幕av久久爽av| 欧美天堂亚洲电影院在线播放| 欧美麻豆精品久久久久久| 综合欧美精品| 午夜剧场免费看| 99中文字幕在线| 欧美r级电影在线观看| 丝袜视频国产在线播放| 欧美777四色影视在线| 日韩亚洲精品视频| 黄色a在线观看| 欧美国产三区| 精品欧美日韩在线| 精品视频全国免费看| 俄罗斯男人又粗又大| 久久久精品国产网站| 亚洲精品国产福利| 中文字幕一区二区三区四区| 少妇大叫太大太粗太爽了a片小说| 韩国成人精品a∨在线观看| 2019中文在线观看| 99t1这里只有精品| 老司机免费视频一区二区| 高清日韩电视剧大全免费| 卡通动漫国产精品| 久久国产毛片| 人人网欧美视频| 亚洲私拍视频| 久草在线视频福利| 亚洲乱妇老熟女爽到高潮的片| 午夜视频久久久久久| 91在线免费看| 第84页国产精品| 欧洲杯半决赛直播| 亚洲国产一成人久久精品| 911精品美国片911久久久| 亚洲成人免费在线观看| 久久精品国产露脸对白| 岛国av在线不卡| 欧美成人家庭影院| 国产精品露脸视频| 国产精品一二三在| 国产精品8888| 欧美黄色网页| 嫩呦国产一区二区三区av| 91精品中文字幕| 人妻偷人精品一区二区三区| 欧美精品七区| 九一在线视频| 亚欧无线一线二线三线区别| 高清欧美性猛交| 黄色小视频在线看| 香蕉影院在线| 午夜欧洲一区| www.亚洲欧美| 免费黄色大片| 亚洲精品第一国产综合精品| 亚洲自拍偷拍一区| 做爰高潮hd色即是空| 亚洲一区二区三区四区不卡| 精品网站在线| 超碰97网站| 日韩和欧美一区二区三区| 女人18毛片水真多18精品| 精品极品三级久久久久| 国产精品久久久久久亚洲毛片| 日韩午夜小视频| 久久全国免费久久青青小草| h视频网站在线观看| av在线播放一区二区| 精人妻无码一区二区三区| 成人久久18免费网站漫画| 思思久久精品视频| 一级全黄裸体片| xxxx日韩| 国产成人强伦免费视频网站| 亚洲第一页视频| 国产精品久久久久久久成人午夜| 四虎成人精品在永久在线观看| 操人视频免费看| av噜噜色噜噜久久| 国产精品欧美激情| 欧美日韩dvd在线观看| 色网站在线播放| 久久久久性色av无码一区二区| 色综合一个色综合亚洲| 精品国产伦一区二区三区观看方式| 久久九九国产精品怡红院| 日韩视频免费| 日韩精品一区二区在线视频| 99国产精品视频免费观看| 超碰超碰超碰超碰超碰| 欧美13videosex性极品| 美女扒开腿让男人桶爽久久软| 色成人在线视频| 日韩中文字幕第一页| 欧美成人精品激情在线观看| 国产精品久久久久久成人| 久久99精品久久只有精品| 猫咪成人官网| 免费在线观看麻豆视频| 国产精选一区二区三区不卡催乳| 好吊色视频一区二区三区| 亚洲综合欧美综合| 成人观看免费视频| 精品久久久国产| 在线heyzo| 欧美丰满嫩嫩电影| 日韩成人综合网| 尤物网站在线看| 日本少妇一级片| 国内精品伊人久久久久影院对白| 国产成人综合一区二区三区| 在线观看的毛片| 天堂网在线免费观看| 高潮久久久久久久久久久久久久| 欧美少妇性生活视频| 91偷拍一区二区三区精品| 欧美精品久久一区二区| 噜噜噜在线观看播放视频| 欧美日韩亚洲综合| 日本特级黄色片| 成人一区二区三区视频在线观看| 久久久久久久久久久久av| 亚洲欧洲激情在线乱码蜜桃| 亚洲成av人综合在线观看| 国产精品国产高清国产| 国产呻吟对白刺激无套视频在线| 白嫩白嫩国产精品| 17婷婷久久www| 天堂av2020| 91视频免费播放| 久久精品免视看| 欧美三区在线视频| 91丨精品丨国产| 男女啪啪999亚洲精品| 91成年人视频| 污污的网站在线免费观看| 久久综合加勒比| 中文乱码字幕av网站| 久久影院理伦片| 午夜高潮免费视频| 日本欧洲一区二区| a v视频在线观看| 中文在线三区| 四虎成人免费电影| 国产精品人妻一区二区三区| 亚洲不卡在线观看| 三级一区在线视频先锋| 粉嫩av一区二区三区四区五区| 色欲一区二区三区精品a片| 暴力调教一区二区三区| 三级精品视频久久久久| 快射视频在线观看| 男人的天堂最新网址| 日日摸天天爽天天爽视频| 最新一本之道波多野结衣| 成人一区二区免费视频| 91九色蝌蚪嫩草| 日本伊人精品一区二区三区观看方式| 日韩乱码人妻无码中文字幕| 神马午夜久久| 成人在线精品视频| porn亚洲| 午夜电影一区二区| 久久国产生活片100|