在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

匯總和深挖C語言中宏定義所有的知識點

FPGA之家 ? 來源:IOT物聯網小鎮 ? 作者:道哥 ? 2021-05-03 18:31 ? 次閱讀

一、前言

二、預處理器的操作

三、宏擴展

四、符號:# 與 ##

五、可變參數的處理

六、奇思妙想的宏

七、總結

一、前言一直以來,我都有這樣一種感覺:當我學習一個新領域的知識時,如果其中的某個知識點在剛開始接觸時,我感覺比較難懂、不好理解,那么以后不論我花多長時間去研究這個知識點,心里會一直認為該知識點比較難,也就是說第一印象特別的重要。

就比如 C 語言中的宏定義,好像跟我犯沖一樣,我一直覺得宏定義是 C 語言中最難的部分,就好比有有些小伙伴一直覺得指針是 C 語言中最難的部分一樣。

宏的本質就是代碼生成器,在預處理器的支持下實現代碼的動態生成,具體的操作通過條件編譯和宏擴展來實現。我們先在心中建立這么一個基本的概念,然后通過實際的描述和代碼來深入的體會:如何駕馭宏定義。

所以,今天我們就來把宏定義所有的知識點進行匯總、深挖,希望經過這篇文章,我能夠擺脫心理的這個魔障??赐赀@篇總結文章后,我相信你也一定能夠對宏定義有一個總體、全局的把握。

二、預處理器的操作1. 宏的生效環節:預處理

一個 C 程序在編譯的時候,從源文件開始到最后生成二進制可執行文件,一共經歷 4 個階段:

32f02058-95c8-11eb-8b86-12bb97331649.png

我們今天討論的內容就是在第一個環節:預處理,由預處理器來完成這個階段的工作,包括下面這 4 項工作:

文件引入(#include);

條件編譯(#if..#elif..#endif);

宏擴展(macro expansions);

行控制(line control)。

2. 條件編譯

一般情況下,C 語言文件中的每一行代碼都是要被編譯的,但是有時候出于對程序代碼優化的考慮,希望只對其中的一部分代碼進行編譯,此時就需要在程序中加上條件,讓編譯器只對滿足條件的代碼進行編譯,將不滿足條件的代碼舍棄,這就是條件編譯。

簡單的說:就是預處理器根據我們設置的條件,對代碼進行動態的處理,把有效的代碼輸出到一個中間文件,然后送給編譯器進行編譯。

條件編譯基本上在所有的項目代碼中都被使用到,例如:當你需要考慮下面的幾種情況時,就一定會使用條件編譯:

需要把程序編譯成不同平臺下的可執行程序;

同一套代碼需要運行在同一平臺上的不同功能產品上;

在程序中存在著一些測試目的的代碼,不想污染產品級的代碼,需要屏蔽掉。

這里舉 3 個例子,在代碼中經常看到的關于條件編譯:

示例1:用來區分 C 和 C++ 代碼

#ifdef __cplusplus extern “C” { #endif void hello(); #ifdef __cplusplus } #endif

這樣的代碼幾乎在每個開源庫中都可能見到,主要的目的就是 C 和 C++ 混合編程,具體來說就是:

如果使用 gcc 來編譯,那么宏 __cplusplus 將不存在,其中的 extern “C” 將會被忽略;

如果使用 g++ 來編譯,那么宏 __cplusplus 就存在,其中的 extern “C” 就發生作用,編譯出來的函數名 hello 就不會被 g++ 編譯器改寫,因此就可以被 C 代碼來調用;

示例2:用來區分不同的平臺

#if defined(linux) || defined(__linux) || defined(__linux__) sleep(1000 * 1000); // 調用 Linux 平臺下的庫函數#elif defined(WIN32) || defined(_WIN32) Sleep(1000 * 1000); // 調用 Windows 平臺下的庫函數(第一個字母是大寫)#endif

那么,這些 linux, __linux, __linux__, WIN32, _WIN32 是從哪里來的呢?我們可以認為是編譯目標平臺(操作系統)為我們預先準備好的。

示例3:在編寫 Windows 平臺下的動態庫時,聲明導出和導入函數

#if defined(linux) || defined(__linux) || defined(__linux__) #define LIBA_API #else #ifdef LIBA_STATIC #define LIBA_API #else #ifdef LIBA_API_EXPORTS #define LIBA_API __declspec(dllexport) #else #define LIBA_API __declspec(dllimport) #endif #endif#endif

// 函數聲明LIBA_API void hello();

這段代碼是直接從我之前在 B 站錄制的一個小視頻里的示例拿過來的,當時主要是演示如何如何在 Linux 平臺下使用 make 和 cmake 構建工具來編譯,后來又小伙伴讓我在 Windows 平臺下也用 make 和 cmake 來構建,所以就寫了上面這段宏定義。

在使用 MSVC 編譯動態庫時,需要在編譯選項(Makefle 或者 CMakeLists.txt)中定義宏 LIBA_API_EXPORTS,那么導出函數 hello 的最前面的宏 LIBA_API 就會被替換成:__declspec(dllexport),表示導出操作;

在編譯應用程序的時候,使用動態庫,需要 include 動態庫的頭文件,此時在編譯選項中不需要定義宏 LIBA_API_EXPORTS,那么 hello 函數最前面的 LIBA_API 就會被替換成 __declspec(dllimport),表示導入操作;

補充一點:如果使用靜態庫,編譯選項中不需要任何宏定義,那么宏 LIBA_API 就為空。

3. 平臺預定義的宏

上面已經看到了,目標平臺會為我們預先定義好一些宏,方便我們在程序中使用。除了上面的操作系統相關宏,還有另一類宏定義,在日志系統中被廣泛的使用:

FILE:當前源代碼文件名;

LINE:當前源代碼的行號;

FUNCTION:當前執行的函數名;

DATE:編譯日期;

TIME:編譯時間;

例如:

printf(“file name: %s, function name = %s, current line:%d

”, __FILE__, __FUNCTION__, __LINE__);

三、宏擴展所謂的宏擴展就是代碼替換,這部分內容也是我想表達的主要內容。宏擴展最大的好處有如下幾點:

減少重復的代碼;

完成一些通過 C 語法無法實現的功能(字符串拼接);

動態定義數據類型,實現類似 C++ 中模板的功能;

程序更容易理解、修改(例如:數字、字符串常亮);

我們在寫代碼的時候,所有使用宏名稱的地方,都可以理解為一個占位符。在編譯程序的預處理環節,這些宏名將會被替換成宏定義中的那些代碼段,注意:僅僅是單純的文本替換。

1. 最常見的宏

為了方便后面的描述,先來看幾個常見的宏定義:

(1) 數據類型的定義

#ifndef BOOL typedef char BOOL;#endif

#ifndef TRUE #define TRUE#endif

#ifndef FALSE #define FALSE#endif

在數據類型定義中,需要注意的一點是:如果你的程序需要用不同平臺下的編譯器來編譯,那么你要去查一下所使用的編譯器對這些宏定義控制的數據類型是否已經定義了。例如:在 gcc 中沒有 BOOL 類型,但是在 MSVC 中,把 BOOL 類型定義為 int 型。

(2) 獲取最大、最小值

#define MAX(a, b) (((a) 》 (b)) ? (a) : (b))#define MIN(a, b) (((a) 《 (b)) ? (a) : (b))

(3) 計算數組中的元素個數

#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))

(4) 位操作

#define BIT_MASK(x) (1 《《 (x))#define BIT_GET(x, y) (((x) 》》 (y)) & 0x01u)#define BIT_SET(x, y) ((x) | (1 《《 (y)))#define BIT_CLR(x, y) ((x) & (~(1 《《 (y))))#define BIT_INVERT(x, y) ((x) ^ (1 《《 (y)))

2. 與函數的區別

從上面這幾個宏來看,所有的這些操作都可以通過函數來實現,那么他們各有什么優缺點呢?

通過函數來實現:

形參的類型需要確定,調用時對參數進行檢查;

調用函數時需要額外的開銷:操作函數棧中的形參、返回值等;

通過宏來實現:

不需要檢查參數,更靈活的傳參;

直接對宏進行代碼擴展,執行時不需要函數調用;

如果同一個宏在多處調用,會增加代碼體積;

還是舉一個例子來說明比較好,就拿上面的比較大小來說吧:

(1) 使用宏來實現

#define MAX(a, b) (((a) 》 (b)) ? (a) : (b))

int main(){ printf(“max: %d

”, MAX(1, 2));}

(2) 使用函數來實現

int max(int a, int b){ if (a 》 b) return a; return b;}

int main(){ printf(“max: %d

”, max(1, 2));}

除了函數調用的開銷,其它看起來沒有差別。這里比較的是 2 個整型數據,那么如果還需要比較 2 個浮點型數據呢?

使用宏來調用:MAX(1.1, 2.2);一切 OK;

使用函數調用:max(1.1, 2.2); 編譯報錯:類型不匹配。

此時,使用宏來實現的優勢就體現出來了:因為宏中沒有類型的概念,調用者傳入任何數據類型都可以,然后在后面的比較操作中,大于或小于操作都是利用了 C 語言本身的語法來執行。

如果使用函數來實現,那么就必須再定義一個用來操作浮點型的函數,以后還有可能比較:char 型、long 型數據等等。

在 C++ 中,這樣的操作可以通過參數模板來實現,所謂的模板也是一種代碼動態生成機制。當定義了一個函數模板后,根據調用者的實參,來動態產生多個函數。例如定義下面這個函數模板:

template《typename T》 T max(T a, T b){ if (a 》 b) return a; return b;}

max(1, 2); // 實參是整型max(1.1, 2,2); // 實參是浮點型

當編譯器看到 max(1, 2) 時,就會動態生成一個函數 int max(int a, int b) { 。.. };

當編譯器看到 max(1.1, 2.2) 時,又會動態生成另一個函數 float max(float a, float b) { 。.. }。

所以,從代碼的動態生成角度看,宏定義和 C++ 中的模板參數有點神似,只不過宏定義僅僅是代碼擴展而已。

下面這個例子也比較不錯,利用宏的類型無關,來動態生成結構體:

#define VEC(T) struct vector_##T { T *data; size_t size; };

int main(){ VEC(int) vec_1 = { .data = NULL, .size = 0 }; VEC(float) vec_2 = { .data = NULL, .size = 0 };}

這個例子中用到了 ##,下面會解釋這個知識點。在前面的例子中,宏的參數傳遞的都是一些變量,而這里傳遞的宏參數是數據類型,通過宏的類型無關性,達到了“動態”創建結構體的目的:

struct vector_int { int *data; size_t size;}

struct vector_float { float *data; size_t size;}

這里有一個陷阱需要注意:傳遞的數據類型中不能有空格,如果這樣使用:VEC(long long),那替換之后得到:

struct vector_long long { // 語法錯誤 long long *data; size_t size;}

四、符號:# 與 ##這兩個符號在編程中的作用也是非常巧妙,夸張的說一句:在任何框架性代碼中,都能見到它們的身影!作用如下:

#:把參數轉換成字符串;

##:連接參數。

1. #: 字符串化

直接看最簡單的例子:

#define STR(x) #xprintf(“string of 123: %s

”, STR(123));

傳入的是一個數字 123,輸出的結果是字符串 “123”,這就是字符串化。

2. ##:參數連接

把宏中的參數按照字符進行拼接,從而得到一個新的標識符,例如:

#define MAKE_VAR(name, no) name##no

int main(void){ int MAKE_VAR(a, 1) = 1; int MAKE_VAR(b, 2) = 2;

printf(“a1 = %d

”, a1); printf(“b2 = %d

”, b2); return 0;}

當調用宏 MAKE_VAR(a, 1) 后,符號 ## 把兩側的 name 和 no 首先替換為 a 和 1,然后連接得到 a1。然后在調用語句中前面的 int 數據類型就說明了 a1 是一個整型數據,最后初始化為 1。

五、可變參數的處理1. 參數名的定義和使用

宏定義的參數個數可以是不確定的,就像調用 printf 打印函數一樣,在定義的時候,可以使用三個點(。..)來表示可變參數,也可以在三個點的前面加上可變參數的名稱。

如果使用三個點(。..)來接收可變參數,那么在使用的時候就需要使用 __VA_ARGS__ 來表示可變參數,如下:

#define debug1(。..) printf(__VA_ARGS__)

debug1(“this is debug1: %d

”, 1);

如果在三個點(。..)的前面加上了一個參數名,那么在使用時就一定要使用這個參數名,而不能使用 __VA_ARGS__ 來表示可變參數,如下:

#define debug2(args.。.) printf(args)

debug1(“this is debug2: %d

”, 2);

2. 可變參數個數為零的處理

看一下這個宏:

#define debug3(format, 。..) printf(format, __VA_ARGS__)

debug3(“this is debug4: %d

”, 4);

編譯、執行都沒有問題。但是如果這樣來使用宏:

debug3(“hello

”);

編譯的時候,會出現錯誤: error: expected expression before ‘)’ token。為什么呢?

看一下宏擴展之后的代碼(__VA_ARGS__為空):

printf(“hello

”,);

看出問題了吧?在格式化字符串的后面多了一個逗號!為了解決問題,預處理器給我們提供了一個方法:通過 ## 符號把這個多余的逗號給自動刪掉。于是宏定義改成下面這樣就沒有問題了。

#define debug3(format, 。..) printf(format, ##__VA_ARGS__)

類似的,如果自己定義了可變參數的名字,也在前面加上 ##,如下:

#define debug4(format, args.。.) printf(format, ##args)

六、奇思妙想的宏宏擴展的本質就是文本替換,但是一旦加上可變參數(__VA_ARGS__)和 ## 的連接功能,就能夠變化出無窮的想象力。

我一直堅信,模仿是成為高手的第一步,只有見多識廣、多看、多學習別人是怎么來使用宏的,然后拿來為己所用,按照“先僵化-再優化-最后固化”這個步驟來訓練,總有一天你也能成為高手。

這里我們就來看幾個利用宏定義的巧妙實現。

1. 日志功能

在代碼中添加日志功能,幾乎是每個產品的標配了,一般見到最普遍的是下面這樣的用法:

#ifdef DEBUG #define LOG(。..) printf(__VA_ARGS__)#else #define LOG(。..) #endif

int main(){ LOG(“name = %s, age = %d

”, “zhangsan”, 20); return 0;}

在編譯的時候,如果需要輸出日志功能就傳入宏定義 DEBUG,這樣就能打印輸出調試信息,當然實際的產品中需要寫入到文件中。如果不需要打印語句,通過把打印日志信息那條語句定義為空語句來達到目的。

換個思路,我們還可以通過條件判斷語句來控制打印信息,如下:

#ifdef DEBUG #define debug if(1)#else #define debug if(0)#endif

int main(){ debug { printf(“name = %s, age = %d

”, “zhangsan”, 20); } return 0;}

這樣控制日志信息的看到的不多,但是也能達到目的,放在這里只是給大家開闊一下思路。

2. 利用宏來迭代每個參數

#define first(x, 。..) #x#define rest(x, 。..) #__VA_ARGS__

#define destructive(。..) do { printf(“first is: %s

”, first(__VA_ARGS__)); printf(“rest are: %s

”, rest(__VA_ARGS__)); } while (0)

int main(void){ destructive(1, 2, 3); return 0;}

主要的思想就是:每次把可變參數 VA_ARGS 中的第一個參數給分離出來,然后把后面的參數再遞歸處理,這樣就可以分離出每一個參數了。我記得侯杰老師在 C++ 的視屏中,利用可變參數模板這個語法,也實現了類似的功能。

剛才在有道筆記中居然找到了侯杰老師演示的代碼,熟悉 C++ 的小伙伴可以研究下下面這段代碼:

// 遞歸的最后一次調用void myprint(){}

template 《typename T, typename.。. Types》void myprint(const T &first, const Types&.。. args){ std::cout 《《 first 《《 std::endl; std::cout 《《 “remain args size = ” 《《 sizeof.。.(args) 《《 std::endl; // 把其他參數遞歸調用 myprint(args.。.);}

int main(){ myprint(“aaa”, 7.5, 100); return 0;}

3. 動態的調用不同的函數

// 普通的枚舉類型enum { ERR_One, ERR_Two, ERR_Three};

// 利用 ## 的拼接功能,動態產生 case 中的比較值,以及函數名。#define TEST(no) case ERR_##no: Func_##no(); break;

void Func_One(){ printf(“this is Func_One

”);}

void Func_Two(){ printf(“this is Func_Two

”);}

void Func_Three(){ printf(“this is Func_Three

”);}

int main(){ int c = ERR_Two; switch (c) { TEST(One); TEST(Two); TEST(Three); };

return 0;}

在這個例子中,核心在于 TEST 宏定義,通過 ## 拼接功能,構造出 case 分支的比較目標,然后動態拼接得到對應的函數,最后調用這個函數。

4. 動態創建錯誤編碼與對應的錯誤字符串

這也是一個非常巧妙的例子,利用了 #(字符串化) 和 ##(拼接) 這 2 個功能來動態生成錯誤編碼碼和相應的錯誤字符串:

#define MY_ERRORS E(TOO_SMALL) E(TOO_BIG) E(INVALID_VARS)

#define E(e) Error_## e,typedef enum { MY_ERRORS} MyEnums;#undef E

#define E(e) #e,const char *ErrorStrings[] = { MY_ERRORS};#undef E

int main(){ printf(“%d - %s

”, Error_TOO_SMALL, ErrorStrings[0]); printf(“%d - %s

”, Error_TOO_BIG, ErrorStrings[1]); printf(“%d - %s

”, Error_INVALID_VARS, ErrorStrings[2]);

return 0;}

我們把宏展開之后,得到一個枚舉類型和一個字符串常量數組:

typedef enum { Error_TOO_SMALL, Error_TOO_BIG, Error_INVALID_VARS,} MyEnums;

const char *ErrorStrings[] = { “TOO_SMALL”, “TOO_BIG”, “INVALID_VARS”,};

宏擴展之后的代碼是不是很簡單啊。編譯、執行結果如下:

0 - TOO_SMALL 1 - TOO_BIG 2 - INVALID_VARS

七、總結有些人對宏愛之要死,多到濫用的程度;而有些人對宏恨之入骨,甚至用上了邪惡(evil)這個詞!其實宏對于 C 來說,就像菜刀對于廚師和歹徒一樣:用的好,可以讓代碼結構簡潔、后期維護特別方便;用的不好,就會引入晦澀的語法、難以調試的 Bug。

對于我們開發人員來說,只要在程序的執行效率、代碼的可維護性上做好平衡就可以了。

原文標題:提高代碼逼格的利器:宏定義-從入門到放棄

文章出處:【微信公眾號:FPGA之家】歡迎添加關注!文章轉載請注明出處。

責任編輯:haq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • C語言
    +關注

    關注

    180

    文章

    7604

    瀏覽量

    136842
  • 代碼
    +關注

    關注

    30

    文章

    4788

    瀏覽量

    68617

原文標題:提高代碼逼格的利器:宏定義-從入門到放棄

文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    《DNESP32S3使用指南-IDF版_V1.6》第二章 常用的C語言知識點

    第二章常用的C語言知識點 本章將為大家介紹常用的C語言知識點。對于已經熟練掌握
    發表于 11-21 09:26

    接口測試理論、疑問收錄與擴展相關知識點

    本文章使用王者榮耀游戲接口、企業微信接口的展示結合理論知識,講解什么是接口測試、接口測試理論、疑問收錄與擴展相關知識點知識學院,快來一起看看吧~
    的頭像 發表于 11-15 09:12 ?316次閱讀
    接口測試理論、疑問收錄與擴展相關<b class='flag-5'>知識點</b>

    C語言中的socket編程基礎

    Socket編程簡介 Socket是一種通信機制,允許程序之間進行通信。在C語言中,socket編程是網絡編程的基礎。通過使用socket,程序可以發送和接收數據,實現不同計算機之間的通信
    的頭像 發表于 11-01 16:51 ?325次閱讀

    C語言中最常見的定義寫法

    如果讓你用C語言寫個定義,我相信大部分同學順手就能寫出define。
    的頭像 發表于 10-28 11:12 ?280次閱讀

    c語言中從左到右結合怎么看

    C語言中,操作符的結合性(Associativity)是指當操作符在表達式中連續出現時,它們如何與操作數結合的順序。對于大多數二元操作符(即需要兩個操作數的操作符),C語言遵循兩種基
    的頭像 發表于 08-20 11:42 ?885次閱讀

    技術干貨驛站 ▏深入理解C語言:基本數據類型和變量

    語言知識,為后續的編程學習打下堅實的基礎。1基本數據類型在C語言中,數據類型指的是用于聲明不同類型的變量或函數的一個廣泛的系統,用于定義
    的頭像 發表于 07-26 17:53 ?2132次閱讀
    技術干貨驛站 ▏深入理解<b class='flag-5'>C</b><b class='flag-5'>語言</b>:基本數據類型和變量

    C++語言基礎知識

    電子發燒友網站提供《C++語言基礎知識.pdf》資料免費下載
    發表于 07-19 10:58 ?7次下載

    模擬電子技術知識點問題總結概覽

    給大家分享模擬電子技術知識點問題總結。
    的頭像 發表于 05-08 15:16 ?1169次閱讀
    模擬電子技術<b class='flag-5'>知識點</b>問題總結概覽

    嵌入式系統中C語言結構體的基礎實現與應用

    C語言中的數組只能允許程序員定義存儲相同類型數據。但是結構是C語言編程中允許您存儲不同數據類型的數據。
    發表于 03-12 14:29 ?502次閱讀
    嵌入式系統中<b class='flag-5'>C</b><b class='flag-5'>語言</b>結構體的基礎實現與應用

    C語言中的typedef的應用

    C 語言提供了 typedef 關鍵字,您可以使用它來為類型取一個新的名字。下面的實例為單字節數字定義了一個術語 BYTE。
    發表于 03-06 11:34 ?387次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言中</b>的typedef的應用

    C語言#define的應用

    C/C++ 編程語言中,當程序被編譯時,被發送到編譯器,編譯器將程序轉換為機器語言,然后完成編譯并執行該程序。預處理器也稱為預處理器。
    發表于 03-06 11:29 ?380次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>#define的應用

    介紹C語言中錯誤處理和異常處理的一些常用的方法和策略

    C語言是一種低級的、靜態的、結構化的編程語言,它沒有提供像C++或Java等高級語言中的異常處理機制,例如try-catch-finally
    的頭像 發表于 02-28 14:25 ?621次閱讀

    C語言中的可變參數介紹

    C 語言為這種情況提供了一個解決方案,它允許您定義一個函數,能根據具體的需求接受可變數量的參數
    發表于 02-28 14:00 ?309次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言中</b>的可變參數介紹

    枚舉有多大?c語言枚舉end的作用是什么?

    枚舉有多大?c語言枚舉end的作用是什么? 枚舉在C語言中是一種常見的數據類型,用于定義一組相互關聯的常量或者變量。它通常用于表示一系列可能
    的頭像 發表于 01-19 14:19 ?599次閱讀

    如何解決C語言中的“訪問權限沖突”異常?C語言引發異常原因分析

    一些措施來解決和防止其發生。本文將詳細介紹C語言中訪問權限沖突異常的原因以及解決方法。 一、訪問權限沖突異常的原因分析 訪問權限沖突異??煞譃閮深悾涸L問私有成員和訪問未定義成員。下面分別分析這兩種異常的原因。 1. 訪問
    的頭像 發表于 01-12 16:03 ?5724次閱讀
    主站蜘蛛池模板: 欧美爽爽网| 一区二区不卡视频在线观看| 欧美一级免费| 日本黄页网站| 午夜欧美在线| 日本三级hd高清电影| 午夜影院在线看| 狠狠操夜夜爱| 亚洲一区二区三区在线| 色五五月五月开| 国产ar高清视频+视频| jiucao在线观看精品| 亚洲国产成人va在线观看| 韩国免费三片在线视频| 久久福利网| 亚州1区2区3区4区产品乱码2021| 国产视频三区| 99草在线视频| 色婷丁香| 天天干天天干天天色| tom影院亚洲国产日本一区| 欧美αv日韩αv另类综合| 免费又爽又黄禁片视频在线播放| 在线a亚洲老鸭窝天堂新地址| 日本拍拍视频| 日本加勒比视频在线观看| 真人实干一级毛片aa免费 | 日韩毛片免费| 大尺度视频网站久久久久久久久| 性欧美精品xxxx| 欧美一区二区三区免费看| 免费欧美黄色| 午夜色综合| 亚色综合| 种子天堂bt| 免费三级毛片| 天天操天天干天天做| 在线欧美色图| 亚洲操操操| 一级做a爰片久久毛片鸭王| 一级片免费在线观看视频|