青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

loop_in_codes

低調做技術__歡迎移步我的獨立博客 codemaro.com 微博 kevinlynx

代碼自動生成-宏遞歸思想

Macro Recursion
author: Kevin Lynx

Preface

    本文可能是<代碼自動生成-宏帶來的奇技淫巧>的續寫。我盡力闡述如何讓宏遞歸(或者說重復)地有規律地產生一
些符號,而讓我們少寫很多重復代碼,也許這些代碼只有那么一點點的不同。將這項小技巧用于底層庫的編寫,會讓代碼
看起來干凈不少,同時文件尺寸也會驟然下降。


Problem


    如果你曾經寫過functor,那么你肯定對某些代碼進行粘貼復制然后修改。更讓人郁悶的是,這些代碼基本是一樣的。
例如,一個典型的functor可能為:

    template <typename Prototype>
    
class functor;
    template 
<typename R, typename P1>
    
class functor<R(P1)>;
    template 
<typename R, typename P1, typename P2>
    
class functor<R(P1,P2)>;


    //好,接下去你可能厭煩了,可能會復制一個帶有兩個參數的functor,然后修改為處理3個參數的。
    這只是一個很簡單的問題。宏不是c++里的東西,本文自然也不是討論各種花哨的模板技術的。如果我之前那篇關于
宏的文章只是讓你去分析問題以及更深層次地認識宏,那么現在我將分享我的這部分思想給你。
    關于上面的問題,我們期待得到這樣的解決方案:

    template <typename R, DEF_PARAM( 2 )>
    
class functor<R( DEF_ARG( 2 ) )>;


    那么,它將自動生成:

    template <typename R, typename P1, typename P2>
    
class functor<R(P1,P2)>


    也就是說,DEF_PARAM(n)這個宏將根據n值自動生成一串符號,例如DEF_PARAM(2)就生成typename P1, typename P2。
同樣,DEF_ARG(n)也會根據參數生成類似于P1,P2,...,Pn的符號串。

思考

    仔細思考下,我們可以看出DEF_PARAM和DEF_ARG這樣的宏具有一種遞歸的特性(其實說成重復可能更合適):每次展
開的內容基本一樣,不斷調用自身直到遇到終止條件。
    那么,我們的目標鎖定于,用宏來實現遞歸。


Pre-Implement

    在開始之前,我需要告訴你一些基本的東西:
    在閱讀一個宏時,你最好按照預處理的處理方式去逐個展開。當我說到展開時,我的意思是把宏替換為宏體。預處理器
展開宏的過程大致為:如果宏參數也是個宏,那么先將宏參數全部展開,再展開該宏;這個時候會掃描展開后的宏,如果
遇到其他宏,則繼續展開。例如有一下宏:

 

#define PI 3.14
#define MUL_PI( n ) n * PI
#define TWO 2


    當我們寫下MUL_PI( TWO )時,預處理發現MUL_PI的參數TWO 是個宏,那么先將TWO展開得到2,然后將2放進宏體展開
得到 2 * PI;預處理器對 2 * PI 進行掃描,發現還有宏PI,于是對PI做展開,得到 2 * 3.14。這個過程是遞歸的。
    但是也有例外,如果MUL_PI對宏參數進行了#或者##,那么該宏參數不會被展開。(參見以前那篇文章吧)
    任何時候,你可以通過以下宏去查看某個宏展開后的樣子,可以方便你調試你的宏:

#define TO_STRING( x ) TO_STRING1( x )
#define TO_STRING1( x ) #x 


    (為什么要寫個TO_STRING1,因為這是為了讓x充分展開,避免上面提到的那個例外)   

    其他規則我會在文中需要的地方提出來。
實現

    就像大部分介紹遞歸函數時候給的例子,這里我也將階乘作為例子。考慮如下典型的階乘函數:

    int fac( int n )
    
{
        
if( n == 1 ) return 1;
        
return n * fac( n - 1 );
    }
 


    其核心部分在于 n * fac( n - 1 ),我們假設我們的宏也可以寫成這樣的的形式:

    #define FAC( n ) n * FAC( n - 1 )


    但是這樣的宏是有問題的:
    當宏被展開時,如果遇到了自身,那么將被處理為一般符號,例如展開FAC( 3 )時,會遇到 FAC( 2 ),那么就把FAC
( 2 )中的FAC當成了一搬符號。
    這樣的限制注定了我們無法讓宏真正地調用自身來實現遞歸。于是,我們不得不寫下以下丑陋的符號,從而去模擬遞
歸的每一次符號調用:

#define FAC_1( n ) 1
#define FAC_2( n ) n * FAC_##(n-1)( n - 1 )
#define FAC_3( n ) n * FAC_##(n-1)( n - 1 ) 


    這系列宏有點別扭(如果你足夠細心),因為我們明顯知道FAC_2返回的將是2,而FAC_3返回的當時6。我們這里只是
模擬,同樣,這使得我們可以把FAC_1作為遞歸的終止條件。
    我們的預想是,當調用FAC_3時,它把n-1的值2合并到FAC_中,從而調用FAC_2,以此類推。
    但是這依然有問題,編譯器會提示‘找不到符號FAC_’。導致這個問題的原因在于:宏展開只是單純的字符替換,是我們
想太多了,預處理器并不會去計算( n - 1 )的值是多少,也就是我們無法得到FAC_2這個宏。

    所以,FAC_3( 3 ) 會被初次替換為 3 * FAC_(3-1)( 3 - 1 )。這個時候編譯器就把FAC_當成了一個普通符號。我們可以
自己定義個FAC_來證明這一點:

 

#define FAC_( n ) T 

 

    那么,FAC_3( 3 )就被替換為 3 * T(3-1)( 3 - 1 )。

    解決這個問題的辦法關鍵在于,讓預處理器自動計算出( n - 1 )。記住,我們解決問題的唯一辦法是:字符替換。
所以,我們可以寫下如下代碼:

 

#define DEC_1 0
#define DEC_2 1
#define DEC_3 2 

#define DEC( n ) DEC_##n 

 

    通過,DEC( n )這個宏,我們可以獲取到一個( n -1 )的數。例如,DEC( 3 )被替換為 DEC_3,繼續替換為 2。

    于是,我們新的FAC系列宏變為:

 

#define FAC_1( n ) 1
#define FAC_2( n ) n * FAC_##DEC( n )( n - 1 )
#define FAC_3( n ) n * FAC_##DEC( n )( n - 1 ) 

 

    不好意思,這樣依然是不正確的!預處理器直接把FAC_和DEC( n )連接成符號了,而不是單個地先處理他們,最后再
合并他們。

    OK,先解決這個問題:先處理FAC_和DEC( n ),再合并他們,而不是先合并他們。要解決這個問題,可以通過第三個宏
來實現:

 

#define CHR( x, y ) x##y 

 

    作為連接兩個符號為一個符號的宏,這個宏顯然是不正確的,因為宏展開還有個規則:如果宏體對宏參數使用了#或##,
那么宏參數不會被展開,也就是說:如果CHR( FAC_, DEC( 3 ) 那么得到的只會是 FAC_DEC( 3 )。通常情況下我們是
再寫個宏:

 

#define CHR( x, y ) CHR1( x, y )
#define CHR1( x, y ) x##y 

 

    從而可以保證在正式連接x和y前,x和y都被完全展開。

    這個時候,我們的FAC系列宏變為:

 

#define FAC_1( n ) 1
#define FAC_2( n ) n * CHR( FAC_, DEC( n ) )( n - 1 )
#define FAC_3( n ) n * CHR( FAC_, DEC( n ) )( n - 1 ) 

 

    結果呢?結果還是有問題。= =
    我們假設CHR( FAC_, DEC( n ) )已經真的按我們預想展開為 FAC_2了,那么FAC_3( 3 )會被展開為什么呢?
被展開為 3 * FAC_2( 3 - 1 )。這是錯誤的,傳給 FAC_2 的參數是 3 - 1就意味著錯誤。我們又臆想預處理器會
幫我們計算 3 - 1的結果了。我們必須保證傳給 FAC_2的參數是個數字2。解決這個問題的辦法就是通過DEC(n)宏。

   于是,FAC系列宏變為:

 

#define FAC_1( n ) 1
#define FAC_2( n ) n * CHR( FAC_, DEC( n ) )( DEC( n ) )
#define FAC_3( n ) n * CHR( FAC_, DEC( n ) )( DEC( n ) ) 

 

    這個時候,FAC_3( 3 )將會被替換為:3*2*1。這就是我們要的結果。

In practice

    以上只是向你展示一個過程,用宏去計算階乘,就像用模板去計算階乘(模板元編程)一樣,只是一個用于展示的東西,
沒有什么實際價值。接下來我們開始有實際的工作,完成之前的預想:

 

template <typename R, typename P1, typename P2, typename P3>
class functor<R (P1, P2, P3)> 

 

    直接:

 

template <typename R, PARAM( 3 )>
class functor<R (ARG( 3 ))> 

 

    先考慮PARAM宏,該宏的任務就是生成類似于:typename P1, typename P2, typename P3這樣的符號。我們假象它每一次
遞歸都生成 typename Pn, 的字符串,那么當他遞歸完時,可能就生成typename P1, typename P2, typename P3, 結果
多了個逗號,也許最后一次結果不該有逗號。

    ARG宏和PARAM宏本質上相同,只是重復的符號不是typename Pn,而是Pn。

    最直接想到的是:

 

#define PARAM_1( n ) typename P##n
#define PARAM_2( n ) CHR( PARAM_, DEC( n ) )( DEC( n ) )##,typename P##n
#define PARAM_3( n ) CHR( PARAM_, DEC( n ) )( DEC( n ) )##,typename P##n 

 

    結果我們得到了個錯誤的展開結果:
typename PDEC( 2 ),typename PDEC( 3 ),typename P3

    這個問題出在:PARAM_3( 3 )當替換為 PARAM_2( DEC( n ) )時,因為PARAM_2(n)宏對于宏參數n使用了##,也就是那個
typename P##n,所以這里不會把 DEC( n )展開,而是直接接到P后面。所以就成了typename PDEC( 3 )。

    為了消除這個問題,我們改進PARAM為:

 

#define TYPE( n ) ,typename P##n
#define PARAM_1( n ) CHR( typename P, n )
#define PARAM_2( n ) CHR( CHR( PARAM_, DEC( n ) )( DEC( n ) ), TYPE( n ) )
#define PARAM_3( n ) CHR( CHR( PARAM_, DEC( n ) )( DEC( n ) ), TYPE( n ) ) 

 

    之所以加入TYPE(n),是因為 ,typename P##n 這個宏本身存在逗號,將其直接用于宏體會出現問題。

    于是,我們得到了正確的結果。

    其實,PARAM系列宏宏體基本是一樣的,除了終止條件那個宏,為什么我們要寫這么多呢?理由在于宏體不能自己調用
自己,所以才有了PARAM_3, PARAM_2。

    我們可以將上面的一系列宏抽象化,使其具有可復用性:

 

#define PARAM( n ) ,typename P##n
#define PARAM_END typename P 

#define ARG( n ) ,P##n
#define ARG_END P 

#define PARAM_1( n ) CHR( typename P, n )
#define PARAM_2( n ) CHR( CHR( PARAM_, DEC( n ) )( DEC( n ) ), TYPE( n ) )
#define PARAM_3( n ) CHR( CHR( PARAM_, DEC( n ) )( DEC( n ) ), TYPE( n ) ) 

#define REPEAT_1( n, f, e ) CHR( e, n )
#define REPEAT_2( n, f, e ) CHR( CHR( REPEAT_, DEC( n ) )( DEC( n ), f, e ), f( n ) )
#define REPEAT_3( n, f, e ) CHR( CHR( REPEAT_, DEC( n ) )( DEC( n ), f, e ), f( n ) ) 

#define DEF_PARAM( n ) REPEAT_##n( n, PARAM, PARAM_END )
#define DEF_ARG( n ) REPEAT_##n( n, ARG, ARG_END ) 

 

    我們創建了可重用的REPEAT系列宏,用于創建諸如typename P1, typename P2, typename P3或者P1,P2,P3之類的符號,
通過更上層的DEF_PARAM(n)和DEF_ARG(n),就可以直接創建出我們上面所需要的符號串,例如:

    DEF_PARAM( 3 ) 就得到 typename P1, typename P2, typename P3
    DEF_ARG( 3 ) 就得到 P1, P2, P3

More in practice

    下載中提供了我使用這個宏遞歸技術寫的lua_binder(如果你看過<實現自己的LUA綁定器-一個模板編程挑戰 >),你
可以與上一個版本做一下比較,代碼少了很多。
    同樣,我希望你也能獲取這種宏遞歸的思想。   

相關下載

   使用宏遞歸的lua_binder

posted on 2008-08-20 17:48 Kevin Lynx 閱讀(12461) 評論(25)  編輯 收藏 引用 所屬分類: c/c++通用編程

評論

# re: 代碼自動生成-宏遞歸思想[未登錄] 2008-08-20 21:39

奇技淫巧還是慎用吧,代碼是寫給人看的,而不是寫給機器看的.  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-20 22:15 沈臻豪(foxtail)

用的好還是很好的,再說賣弄賣弄也挺好。呵呵@創
  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想[未登錄] 2008-08-20 23:07

@沈臻豪(foxtail)
你賣弄完了,接手或者閱讀你代碼的人會吐的.
  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-20 23:24 cexer

有的時候宏的確實很有用,BOOST里面有一個預處理元編程庫。  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-20 23:27 cexer

preprocessor,預處理元編程工具,包含重復和遞歸, 作者 Vesa Karvonen 和 Paul Mensonides.  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-20 23:58 x-matrix

@創
各種技術都有一定的應用場合的,復雜的技術也是。  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-21 10:38 陳梓瀚(vczh)

仍然沒有解決問題。真正的遞歸應當使得你那個PARAM(n)的n不受限制。不過這也沒辦法,是語法本身的限制。  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-22 11:17 空明流轉

很多任務我現在都用Code Generator做了。
用宏太多其實并不好。
不過Code Generator一般需要添加一些預編譯事件什么的  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-08-24 15:55 戴爾1420

通過宏遞歸有規律地產生一些符號,來讓我們少寫很多重復代碼,的確很新穎。  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-12 20:27 littlewater

很好,稍微改改可以支持更多的東西=3=
另外有一點可惜,MACRO好像不能夠處理另外一個MACRO(n)的連續迭代,因為另外一個MACRO展開以后可能會干擾正常的分析比如一個函數……  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-14 13:12 littlewater

再改進了一下,如果是2005以上就可以任意了因為可以用...的MACRO處理……但是高興了半天以后才發現自家用的2003沒機會碰...立馬狂吐錯誤……偶也吐了。。。。

ToT如果有很好的解決方案請告訴偶,謝謝了ToT,實在想不出來了

例子,比如第一輪使用了LZ的宏以后就會出現
template <typename R, DEF_PARAM( 1 )>
class functor<R( DEF_ARG( 1 ) )>;
template <typename R, DEF_PARAM( 2 )>
class functor<R( DEF_ARG( 2 ) )>;
template <typename R, DEF_PARAM( 3 )>
class functor<R( DEF_ARG( 3 ) )>;
顯然可以繼續用迭代的,但是問題在于這些里面有,這個符號= =!!
有解決辦法么??  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-16 16:10 Kevin Lynx

#define PARAM( n ) ,typename P##n
#define PARAM_END typename P1

去掉那個逗號不就可以了?

有些不明白littlewater意思。  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-16 17:54 littlewater

不是這個意思,因為第一次在使用過宏遞歸以后:
template <typename R, DEF_PARAM( 1 )>
class functor<R( DEF_ARG( 1 ) )>;
會被展開為理想的代碼:
template <typename R, typename P1>
class functor<R(P1)>;
問題是這一輪繼續被遞歸的話:

DEF_XXX( template <typename R, typename P1> class functor<R(P1)>; )
實際上多了一個typename R的逗號,應該會被誤判的,結果參數變多了,明白了吧?  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-16 17:55 littlewater

怎么去掉那個逗號呢??  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-17 09:52 Kevin Lynx

@littlewater
依然不明白什么是“這一輪繼續被遞歸”,更不明白你寫下的
“DEF_XXX( template <typename R, typename P1> class functor<R(P1)>; )

是為了說明什么。

我推測,你的意思是說,當宏參數本身也是一個宏,而這個宏的宏體內有逗號時,將會出現歧義:
#define PARAM typename P1,
#define DEF_PARAM( a, b ) something

DEF_PARAM( PARAM, something ); 時,在展開宏體時就會出現DEF_PARAM( typename P1, , something ) 就會出現兩個逗號。

解決這個問題的辦法時,不讓PARAM宏提前展開。

宏展開的一個規則是:如果某個宏(如DEF_PARAM)的宏實參也是一個宏(如PARAM),那么在展開這個宏之前,會先展開宏實參,并將展開后的宏體替換到宏中,然后第二次掃描,如果還有宏,則繼續展開。

所以,解決辦法就是,讓實參不是一個宏!

宏展開還有一個規則是:即使宏實參是一個宏,但是這個宏具有括號屬性,例如
#define PARAM( n ) ,typename P##n 中PARAM宏就是這么一個具有括號屬性的宏,該宏作為宏實參時,如果沒有提供其參數,那么它將被作為普通符號,而不是一個宏。

因此,在代碼kl_macro_params.h中:
#define PARAM( n ) ,typename P##n
#define DEF_PARAM( n ) REPEAT_##n( n, PARAM, PARAM_END )

若DEF_PARAM( 2 ) 時,會得到REPEAT_2( 2, PARAM, PARAM_END )展開REPEAT_2宏時,并不會先展開PARAM,因為PARAM是一個具有括號屬性的宏,如果展開,那么將出現你說的問題。
  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-17 15:55 littlewater

不知道是不是 有點搞糊涂說~
偶的意思是,首先受用KL的方法實現:
template <typename R, DEF_PARAM( 1 )>
class functor<R( DEF_ARG( 1 ) )>;
template <typename R, DEF_PARAM( 2 )>
class functor<R( DEF_ARG( 2 ) )>;
template <typename R, DEF_PARAM( 3 )>
class functor<R( DEF_ARG( 3 ) )>;
就是說,我已經通過 KL系列宏遞歸 定義得到了三組functor的特化;
我現在呢,希望能夠繼續使用 KL系列宏遞歸,把這三個變化得到
#define TINST(n) \
template <typename R, DEF_PARAM( n )> \
class functor<R( DEF_ARG( n ) )>
接著得到:
TINST(1); TINST(2); TINST(3); ...
想能不能再用一次宏遞歸(就是所謂的再一次,連續遞歸的意思~)
#define DEF_TINST(n) REPEAT_##n( n, TINST1, TINST_END )
當然TINST1和END分別是TINST(1)和TINST(n)了

然后只需要調用一次DEF_TINST(3);就完成了所有的任務了,增加或者減少一個定義只需要修改3為其他的數值,不需要再增加其他的代碼了

以上偶滴愿望,不知道是否說得明白?

然后實際上似乎有問題,8過不排除敝人的異常=v=+  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-17 17:35 littlewater

仔細再看了一遍KL的文章開頭和KL最后的那個回復,總算明白問題所在=3=
實際上歸納了一下問題是宏自身調用:
#define D(f,n) f(n)
#define A(n) B(n), D(B,n+1)
#define B(x) 1
int main( void )
{
int x[] = { D(A, n) };
return 0;
}
原來宏不能夠解開自身,由于REPEAT_xx偶在連續遞歸的時候重復調用了自身OTL,因為比較龐大失誤了,而且對宏不熟悉——
現在好了,也不需要考慮是2005了——感謝。。。
  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想[未登錄] 2008-09-24 11:01 Kevin Lynx

今天發現boost果然有這么一個宏庫:
http://www.boost.org/doc/libs/1_36_0/libs/preprocessor/doc/index.html

然后在<C++ Template Metaprogramming>一書里也看到類似的闡述:
http://www.boostpro.com/tmpbook/preprocessor.html

原來我又重造了一次輪子,還沒造好。 = =|
  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2008-09-24 15:53 littlewater

不,很不錯了,現在也可以比較一下,看看差距=3=~  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2009-11-23 09:43 Scan

我感覺你那個根本不是遞歸啊
既然宏的部分也得編碼, 那為何不這樣?

#define PARAM_1(t) t T
#define PARAM_2(t) PARAM_1(t), t T2
#define PARAM_3(t) PARAM_2(t), t T3

#define PARAM(n, t) PARAM_##n(t)

#define TO_STRING1(x) #x
#define TO_STRING(x) TO_STRING1(x)

int main()
{
cout << TO_STRING(PARAM(3, typename)) << endl;
cout << TO_STRING(PARAM(3, )) << endl;
}  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2011-01-13 00:03 溪流

有沒有辦法只出現一次 DEX_XX 這樣的系列宏,
之后不要出現 PARAM_1 PARAM_2 PARAM_3 呢?  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2011-01-13 09:10 Kevin Lynx

@溪流
從原理上來看的話,我們最終需要的是各種帶有1、2、3之類的相似符號,例如 typename P1,typename P2,....,所以,逐步拆分這些符號后,就會自然而然地得到”基礎數字序列生成器”:DEC。

相當于,DEC系列宏就是這個宏庫的基礎,而PARAM_1則算是稍微上層一點的應用。

ps,很久沒搗鼓這些復雜的東西,諸多遺忘,見諒。  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2011-07-07 23:19 enjoylife

考慮要用宏計算一組100至1000的階層,不能寫901行FAC_3( n )的調用吧。寫成for(int i=100;i<=1000;i++) FAC_3(i)又不行。怎么辦?  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2011-07-07 23:31 enjoylife

這組計算階層的宏,只能計算到3的階層...考慮有什么辦法可以往上加?  回復  更多評論   

# re: 代碼自動生成-宏遞歸思想 2015-05-08 20:04 邊城浪

有意思. 非常棒  回復  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            在线观看精品一区| 亚洲最新视频在线播放| 一区二区日韩欧美| 国语精品一区| 精品成人国产在线观看男人呻吟| 欧美黄色免费| 欧美日韩日本国产亚洲在线| 欧美顶级艳妇交换群宴| 欧美成人免费网站| 亚洲国产精品精华液网站| 欧美一区二区三区四区视频| 欧美一区二区福利在线| 香蕉成人啪国产精品视频综合网| 久久成人精品| 久久亚洲私人国产精品va| 欧美国产高潮xxxx1819| 欧美激情久久久久| 一本色道久久综合狠狠躁篇的优点| 欧美黄色成人网| 99国产精品久久久久老师| 一区二区三区四区蜜桃| 久久成人这里只有精品| 久久久国产一区二区| 欧美电影免费观看网站| 欧美人与性禽动交情品| 欧美精品一区二区三| 国产精品一区二区三区四区五区| 国产美女一区二区| 91久久久久久国产精品| av成人激情| 母乳一区在线观看| 久久综合伊人77777| 欧美午夜精品久久久久久浪潮| 欧美色图天堂网| 国产一区二区三区久久悠悠色av | 久久亚洲国产成人| 国产精品久久久久久久久久妞妞 | 精品动漫3d一区二区三区| 亚洲伊人一本大道中文字幕| 久久激情五月激情| 99xxxx成人网| 久久另类ts人妖一区二区| 亚洲成色777777在线观看影院| 99精品热视频只有精品10| 99国产精品99久久久久久| 欧美粗暴jizz性欧美20| 国产精品亚洲综合天堂夜夜| 在线亚洲免费视频| 欧美成人视屏| 久久人人看视频| 国产精品欧美日韩一区| 99视频精品| 欧美电影免费观看| 老司机一区二区| 国产亚洲精品aa午夜观看| 亚洲欧美日韩在线| 亚洲精品免费一二三区| 欧美1区3d| 国内精品国语自产拍在线观看| 午夜精品一区二区三区电影天堂| 免费观看成人| 久久国产精彩视频| 国产精品视频久久一区| 亚洲视频在线二区| 亚洲精品在线观看免费| 欧美福利影院| 亚洲国产精品尤物yw在线观看 | 欧美日韩一区综合| 亚洲少妇一区| 亚洲人被黑人高潮完整版| 久久一区二区三区国产精品| 国产欧美日韩在线播放| 欧美一级日韩一级| 亚洲欧美日韩区| 欧美日韩在线第一页| 亚洲精选视频在线| 91久久久久久国产精品| 欧美国产日韩一区二区在线观看| 日韩一级在线| 亚洲国产精品t66y| 国产精品久久精品日日| 亚洲一区免费看| 免费不卡在线观看| 久久国产婷婷国产香蕉| 在线播放日韩专区| 久久伊伊香蕉| 欧美精品色网| 一本色道久久综合亚洲二区三区| 在线性视频日韩欧美| 国产精品亚洲一区二区三区在线| 久久一区二区三区四区| 久久亚洲精品一区| 亚洲一级片在线观看| 亚洲一区二区三区精品在线| 韩国在线一区| 亚洲成色777777女色窝| 欧美天堂亚洲电影院在线播放| 午夜精品美女久久久久av福利| 久久成人亚洲| 亚洲国产综合视频在线观看| 一区二区三区av| 国产视频一区免费看| 亚洲国产精品久久| 国产精品久久国产精品99gif | 午夜精品福利一区二区三区av| 香蕉久久夜色精品| 99精品福利视频| 在线亚洲欧美视频| 亚洲国产精品福利| 宅男精品视频| 99精品欧美一区二区三区| 亚洲一区网站| 亚洲图片欧美一区| 日韩视频在线免费| 亚洲高清视频一区| 欧美一区二区在线播放| 日韩视频一区| 久久青草欧美一区二区三区| 亚洲小少妇裸体bbw| 久久国产天堂福利天堂| 亚洲少妇自拍| 另类av一区二区| 久久噜噜噜精品国产亚洲综合 | 这里只有精品丝袜| 亚洲激情二区| 欧美一区二区三区免费在线看| 亚洲毛片在线| 在线天堂一区av电影| 亚洲高清久久网| 久久久亚洲人| 亚洲乱亚洲高清| 欧美国产精品va在线观看| 免费日韩视频| 国产日韩欧美在线一区| 午夜精品免费在线| 亚洲一区二区三区国产| 欧美视频在线观看一区| 久久久精品欧美丰满| 国产精品一二一区| 亚洲综合清纯丝袜自拍| 在线综合亚洲欧美在线视频| 欧美—级在线免费片| 麻豆精品在线播放| 亚洲激情欧美激情| 麻豆精品视频在线观看| 国产一区二区三区四区| 性娇小13――14欧美| 午夜精品国产| 国产拍揄自揄精品视频麻豆| 亚洲美女av黄| 欧美亚洲一区二区在线观看| 老鸭窝毛片一区二区三区| 亚洲高清av在线| 亚洲精品在线观| 欧美大色视频| 亚洲天堂网站在线观看视频| 亚洲一线二线三线久久久| 国产精品视频导航| 亚洲欧美在线一区二区| 久久偷看各类wc女厕嘘嘘偷窃| 国产一区二区三区在线播放免费观看| 久久精品国产亚洲aⅴ| 久久午夜电影| 洋洋av久久久久久久一区| 欧美日韩综合视频| 在线视频你懂得一区二区三区| 欧美在线亚洲| 亚洲第一页自拍| 欧美日韩在线观看视频| 亚洲一区不卡| 欧美aⅴ99久久黑人专区| 亚洲第一主播视频| 国产精品美女www爽爽爽视频| 欧美ed2k| 午夜天堂精品久久久久| 永久91嫩草亚洲精品人人| 欧美va亚洲va香蕉在线| 亚洲系列中文字幕| 久久激情综合网| 99re视频这里只有精品| 国产精品成人在线观看| 久久影音先锋| 亚洲精品久久久久久下一站 | 亚洲欧美成人| 一区二区三区亚洲| 午夜精品国产更新| 亚洲国产精品尤物yw在线观看| 午夜欧美理论片| 亚洲国产中文字幕在线观看| 国产精品一区视频网站| 久久国产乱子精品免费女 | 免费观看日韩| 午夜精品在线| 亚洲经典三级| 一区在线影院| 国产精品高潮视频| 欧美精选一区| 久久影音先锋| 亚洲特级片在线| 99re热精品|