日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

ACE - Reactor模式源码剖析及具体实现(大量源码慎入)

發(fā)布時(shí)間:2025/3/21 编程问答 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 ACE - Reactor模式源码剖析及具体实现(大量源码慎入) 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

原文出自http://www.cnblogs.com/binchen-china,禁止轉(zhuǎn)載。

在之前的文章中提到過(guò)Reactor模式和Preactor模式,現(xiàn)在利用ACE的Reactor來(lái)實(shí)現(xiàn)一個(gè)基于Reactor框架的服務(wù)器。

首先回顧下Reactor模式和Preactor模式。

Reactor模式:

Reactor模式實(shí)現(xiàn)非常簡(jiǎn)單,使用同步IO模型,即業(yè)務(wù)線程處理數(shù)據(jù)需要主動(dòng)等待或詢問(wèn),主要特點(diǎn)是利用epoll監(jiān)聽(tīng)listen描述符是否有響應(yīng),及時(shí)將客戶連接信息放于一個(gè)隊(duì)列,epoll和隊(duì)列都是在主進(jìn)程/線程中,由子進(jìn)程/線程來(lái)接管描述符傳輸?shù)臄?shù)據(jù),對(duì)描述符進(jìn)行下一步操作,包括connect和數(shù)據(jù)讀寫。主程讀寫就緒事件。整個(gè)過(guò)程都需要先獲取描述符狀態(tài),在狀態(tài)允許下再執(zhí)行任務(wù)。

大致流程圖如下:

Preactor模式:

Preactor模式完全將IO處理和業(yè)務(wù)分離,使用異步IO模型,即內(nèi)核完成數(shù)據(jù)處理后主動(dòng)通知給應(yīng)用處理,主進(jìn)程/線程不僅要完成listen任務(wù),還需要完成內(nèi)核數(shù)據(jù)緩沖區(qū)的映射,直接將數(shù)據(jù)buff傳遞給業(yè)務(wù)線程,業(yè)務(wù)線程只需要處理業(yè)務(wù)邏輯即可。整個(gè)過(guò)程直接推送任務(wù),描述符狀態(tài)是否允許執(zhí)行任務(wù)由內(nèi)核去調(diào)度處理。

大致流程如下:

?

ACE的Reactor模式

所有服務(wù)器都可以歸納為以下三層:

  • I/O:處理底層IO事件
  • Dispatch:事件消息派發(fā)
  • Service:業(yè)務(wù)處理

?

ACE的Reactor處于I/O和Dispatch層。提供了I/O監(jiān)控和消息Dispatch。其中I/O需要用戶以handle的形式提供到ACE_Reactor內(nèi)。

Dispatch需要以ACE_Event_Handler為載體,也就是說(shuō)要實(shí)現(xiàn)一個(gè)完整的Reactor只依賴ACE_Reactor類是無(wú)法完成的。

?

上篇博文利用ACE的Socket可以看出一個(gè)ACE_SOCK_Acceptor和ACE_SOCK_Stream就可以完成服務(wù)器代碼。現(xiàn)在要做的是,

1.引入Reactor,把Acceptor和Stream兩個(gè)I/O分別放在兩個(gè)繼承于ACE_Event_Handler的類中注冊(cè)給ACE_Reactor。

2.主函數(shù)注冊(cè)包含ACE_SOCK_Acceptor的類到ACE_Reactor中,當(dāng)ACE_SOCK_Acceptor收到數(shù)據(jù)即有客戶端連接后再給對(duì)應(yīng)的客戶端創(chuàng)建一個(gè)ACE_SOCK_Stream通道并注冊(cè)到ACE_Reactor中。

?

使用ACE_Reactor實(shí)現(xiàn)的Server代碼:

1 #include <ace/INET_Addr.h> 2 #include <ace/SOCK_Acceptor.h> 3 #include <ace/SOCK_Stream.h> 4 #include <ace/Reactor.h> 5 #include <ace/Log_Msg.h> 6 #include <list> 7 8 #define MAX_BUFF_SIZE 1024 9 #define LISTEN_PORT 5010 10 #define SERVER_IP ACE_LOCALHOST 11 12 class ServerStream : public ACE_Event_Handler 13 { 14 public: 15 ServerStream(); 16 ~ServerStream(); 17 ACE_SOCK_Stream& GetStream(){return Svr_stream;} //給accept提供接口綁定數(shù)據(jù)通道 18 virtual int handle_input(ACE_HANDLE fd); //I/O觸發(fā)事件后調(diào)用 19 void close(); 20 virtual ACE_HANDLE get_handle(void) const {return Svr_stream.get_handle();} //不重載需要手動(dòng)將handle傳入ACE_Reactor 21 private: 22 ACE_INET_Addr Cli_addr; 23 ACE_SOCK_Stream Svr_stream; 24 }; 25 26 ServerStream::ServerStream() 27 { 28 29 } 30 31 ServerStream::~ServerStream() 32 { 33 close(); 34 } 35 36 int ServerStream::handle_input(ACE_HANDLE fd) 37 { 38 char strBuffer[MAX_BUFF_SIZE]; 39 int byte = Svr_stream.recv(strBuffer,MAX_BUFF_SIZE); //可讀數(shù)據(jù) 40 if (-1 == byte) 41 { 42 ACE_DEBUG((LM_INFO, ACE_TEXT("receive data failed\n"))); 43 } 44 else if(0 == byte) 45 { 46 close(); 47 ACE_DEBUG((LM_INFO, ACE_TEXT("client closed!\n"))); 48 } 49 else 50 { 51 ACE_DEBUG((LM_INFO, ACE_TEXT("receive from client: %s\n"),strBuffer)); 52 } 53 } 54 55 void ServerStream::close() 56 { 57 Svr_stream.close(); 58 ACE_Reactor::instance()->remove_handler(this,ACE_Event_Handler::READ_MASK | ACE_Event_Handler::DONT_CALL); 59 //delete this; 60 } 61 62 class ServerAcceptor : public ACE_Event_Handler 63 { 64 public: 65 ServerAcceptor(int port,char* ip); 66 ~ServerAcceptor(); 67 bool open(); 68 virtual int handle_input(ACE_HANDLE fd); //有client連接 69 void close(); 70 virtual ACE_HANDLE get_handle(void) const {return Svr_aceept.get_handle();} 71 private: 72 ACE_INET_Addr Svr_addr; 73 ACE_SOCK_Acceptor Svr_aceept; 74 std::list<ServerStream*> m_streamPool; //stream pool 75 }; 76 77 ServerAcceptor::ServerAcceptor(int port,char* ip):Svr_addr(port,ip) 78 { 79 if (!open()) //open listen port 80 { 81 ACE_DEBUG((LM_INFO, ACE_TEXT("open failed!\n"))); 82 } 83 else 84 { 85 ACE_DEBUG((LM_INFO, ACE_TEXT("open success!\n"))); 86 } 87 } 88 89 ServerAcceptor::~ServerAcceptor() 90 { 91 close(); 92 std::list<ServerStream*>::iterator it; 93 for (it = m_streamPool.begin();it != m_streamPool.end();++it) 94 { 95 if (NULL != (*it)) 96 { 97 (*it)->close(); 98 delete (*it); 99 } 100 } 101 } 102 103 bool ServerAcceptor::open() 104 { 105 if (-1 == Svr_aceept.open(Svr_addr,1)) 106 { 107 ACE_DEBUG((LM_ERROR,ACE_TEXT("failed to accept\n"))); 108 Svr_aceept.close(); 109 return false; 110 } 111 return true; 112 } 113 114 int ServerAcceptor::handle_input(ACE_HANDLE fd ) 115 { 116 ServerStream *stream = new ServerStream(); //產(chǎn)生新通道 117 if (NULL != stream) 118 { 119 m_streamPool.push_back(stream); 120 } 121 if (Svr_aceept.accept(stream->GetStream()) == -1) //綁定通道 122 { 123 printf("accept client fail\n"); 124 return -1; 125 } 126 ACE_Reactor::instance()->register_handler(stream,ACE_Event_Handler::READ_MASK); //通道注冊(cè)到ACE_Reactor 127 ACE_DEBUG((LM_ERROR,ACE_TEXT("User connect success!\n"))); 128 } 129 130 void ServerAcceptor::close() 131 { 132 ACE_Reactor::instance()->remove_handler(this,ACE_Event_Handler::ACCEPT_MASK); 133 Svr_aceept.close(); 134 } 135 136 int ACE_TMAIN() 137 { 138 ServerAcceptor server(LISTEN_PORT,(char *)SERVER_IP); 139 ACE_Reactor::instance()->register_handler(&server,ACE_Event_Handler::ACCEPT_MASK); //listen port注冊(cè)到ACE_Reactor 140 ACE_Reactor::instance()->run_reactor_event_loop(); //進(jìn)入消息循環(huán),有I/O事件回調(diào)handle_input 141 return 0; 142 }

測(cè)試結(jié)果:

終端1:

終端2:

終端3:

ACE_Reactor內(nèi)部已經(jīng)幫我們實(shí)現(xiàn)了IO復(fù)用。

?

有了Reactor的demo后,下面一步步查看ACE_Reactor內(nèi)部是如何運(yùn)作的:

ACE_Reactor注冊(cè)EVENT,重載了一個(gè)register_handler:

1 int 2 ACE_Reactor::register_handler (ACE_Event_Handler *event_handler, 3 ACE_Reactor_Mask mask) 4 { 5 // Remember the old reactor. 6 ACE_Reactor *old_reactor = event_handler->reactor (); 7 8 // Assign *this* <Reactor> to the <Event_Handler>. 9 event_handler->reactor (this); 10 11 int result = this->implementation ()->register_handler (event_handler, mask); 12 if (result == -1) 13 // Reset the old reactor in case of failures. 14 event_handler->reactor (old_reactor); 15 16 return result; 17 }

第11行實(shí)際是ACE_Reactor_Impl *implementation (void) const;在做實(shí)際功能。進(jìn)一步查看implementation 是如何注冊(cè)的。

1 template <class ACE_SELECT_REACTOR_TOKEN> int 2 ACE_Select_Reactor_T<ACE_SELECT_REACTOR_TOKEN>::register_handler 3 (ACE_Event_Handler *handler, 4 ACE_Reactor_Mask mask) 5 { 6 ACE_TRACE ("ACE_Select_Reactor_T::register_handler"); 7 ACE_MT (ACE_GUARD_RETURN (ACE_SELECT_REACTOR_TOKEN, ace_mon, this->token_, -1)); 8 return this->register_handler_i (handler->get_handle (), handler, mask); 9 }

這里開(kāi)始大量使用模板,這里重載了兩個(gè)在最后調(diào)用register_handler_i,在第8行,可以看到調(diào)用了get_handle,也就是我們重載的那個(gè)函數(shù),所以我們不需要傳入ACE_Reactor,它在這一步調(diào)用了我們重新的虛函數(shù),獲得了handle,當(dāng)然也可以不做重寫,手動(dòng)傳入handle。這個(gè)handle就是我們要處理的I/O,而handler則是我們繼承ACE_Event_Handler的類。

1 template <class ACE_SELECT_REACTOR_TOKEN> int 2 ACE_Select_Reactor_T<ACE_SELECT_REACTOR_TOKEN>::register_handler_i 3 (ACE_HANDLE handle, 4 ACE_Event_Handler *event_handler, 5 ACE_Reactor_Mask mask) 6 { 7 ACE_TRACE ("ACE_Select_Reactor_T::register_handler_i"); 8 9 // Insert the <handle, event_handle> tuple into the Handler 10 // Repository. 11 return this->handler_rep_.bind (handle, event_handler, mask); 12 }

到這里,我們看到代碼handler_rep_將hande和event_handler綁定了起來(lái),handler_rep_在Select_Reactor_Base.h內(nèi)為ACE_Select_Reactor_Impl的成員變量。下面我們繼續(xù)看bind實(shí)際是在做什么操作。

1 // Bind the <ACE_Event_Handler *> to the <ACE_HANDLE>. 2 int 3 ACE_Select_Reactor_Handler_Repository::bind (ACE_HANDLE handle, 4 ACE_Event_Handler *event_handler, 5 ACE_Reactor_Mask mask) 6 { 7 ACE_TRACE ("ACE_Select_Reactor_Handler_Repository::bind"); 8 9 if (event_handler == 0) 10 return -1; 11 12 if (handle == ACE_INVALID_HANDLE) 13 handle = event_handler->get_handle (); 14 15 if (this->invalid_handle (handle)) 16 return -1; 17 18 // Is this handle already in the Reactor? 19 bool existing_handle = false; 20 21 #if defined (ACE_WIN32) 22 23 map_type::ENTRY * entry = 0; 24 25 int const result = 26 this->event_handlers_.bind (handle, event_handler, entry); 27 28 if (result == -1) 29 { 30 return -1; 31 } 32 else if (result == 1) // Entry already exists. 33 { 34 // Cannot use a different handler for an existing handle. 35 if (event_handler != entry->item ()) 36 { 37 return -1; 38 } 39 else 40 { 41 // Remember that this handle is already registered in the 42 // Reactor. 43 existing_handle = true; 44 } 45 } 46 47 #else 48 49 // Check if this handle is already registered. 50 ACE_Event_Handler * const current_handler = 51 this->event_handlers_[handle]; 52 53 if (current_handler) 54 { 55 // Cannot use a different handler for an existing handle. 56 if (current_handler != event_handler) 57 return -1; 58 59 // Remember that this handle is already registered in the 60 // Reactor. 61 existing_handle = true; 62 } 63 64 this->event_handlers_[handle] = event_handler; 65 66 if (this->max_handlep1_ < handle + 1) 67 this->max_handlep1_ = handle + 1; 68 69 #endif /* ACE_WIN32 */ 70 71 if (this->select_reactor_.is_suspended_i (handle)) 72 { 73 this->select_reactor_.bit_ops (handle, 74 mask, 75 this->select_reactor_.suspend_set_, 76 ACE_Reactor::ADD_MASK); 77 } 78 else 79 { 80 this->select_reactor_.bit_ops (handle, 81 mask, 82 this->select_reactor_.wait_set_, 83 ACE_Reactor::ADD_MASK); 84 85 // Note the fact that we've changed the state of the <wait_set_>, 86 // which is used by the dispatching loop to determine whether it can 87 // keep going or if it needs to reconsult select(). 88 // this->select_reactor_.state_changed_ = 1; 89 } 90 91 // If new entry, call add_reference() if needed. 92 if (!existing_handle) 93 event_handler->add_reference (); 94 95 return 0; 96 }

這里非常關(guān)鍵,第50行,event_handlers_實(shí)則是一個(gè)容器,handle和event_hander以index的方式綁定了起來(lái),存儲(chǔ)在了一個(gè)容器內(nèi),第80行還有一行關(guān)鍵代碼,ADD_MASK形式的操作加入到了wait_set_成員內(nèi)。

注冊(cè)的代碼到這里為止,ACE_Reactor實(shí)際上調(diào)用了幾層N個(gè)文件,其實(shí)就是把handle,即I/O和handler,即繼承ACE_Event_Handler的類綁定在了一個(gè)容器里。下面看ACE_Reactor是如何進(jìn)行消息循環(huán)的。

1 int 2 ACE_Reactor::run_reactor_event_loop (REACTOR_EVENT_HOOK eh) 3 { 4 ACE_TRACE ("ACE_Reactor::run_reactor_event_loop"); 5 6 if (this->reactor_event_loop_done ()) 7 return 0; 8 9 while (1) 10 { 11 int const result = this->implementation_->handle_events (); 12 13 if (eh != 0 && (*eh)(this)) 14 continue; 15 else if (result == -1 && this->implementation_->deactivated ()) 16 return 0; 17 else if (result == -1) 18 return -1; 19 } 20 21 ACE_NOTREACHED (return 0;) 22 }

同樣,將loop交給了ACE_Reactor_Impl *implementation (void) const;操作。

繼續(xù)跟蹤

1 template <class ACE_SELECT_REACTOR_TOKEN> int 2 ACE_Select_Reactor_T<ACE_SELECT_REACTOR_TOKEN>::handle_events 3 (ACE_Time_Value &max_wait_time) 4 { 5 ACE_TRACE ("ACE_Select_Reactor_T::handle_events"); 6 7 return this->handle_events (&max_wait_time); 8 }

再次到了模板,調(diào)用handle_events,下面到了關(guān)鍵代碼

1 template <class ACE_SELECT_REACTOR_TOKEN> int 2 ACE_Select_Reactor_T<ACE_SELECT_REACTOR_TOKEN>::handle_events_i 3 (ACE_Time_Value *max_wait_time) 4 { 5 int result = -1; 6 7 ACE_SEH_TRY 8 { 9 // We use the data member dispatch_set_ as the current dispatch 10 // set. 11 12 // We need to start from a clean dispatch_set 13 this->dispatch_set_.rd_mask_.reset (); 14 this->dispatch_set_.wr_mask_.reset (); 15 this->dispatch_set_.ex_mask_.reset (); 16 17 int number_of_active_handles = 18 this->wait_for_multiple_events (this->dispatch_set_, 19 max_wait_time); 20 21 result = 22 this->dispatch (number_of_active_handles, 23 this->dispatch_set_); 24 } 25 ACE_SEH_EXCEPT (this->release_token ()) 26 { 27 // As it stands now, we catch and then rethrow all Win32 28 // structured exceptions so that we can make sure to release the 29 // <token_> lock correctly. 30 } 31 32 return result; 33 }

第18行wait_for_multiple_events和第22行dispatch。分別做了兩件非常關(guān)鍵的事。

1 // Must be called with lock held. 2 3 template <class ACE_SELECT_REACTOR_TOKEN> int 4 ACE_Select_Reactor_T<ACE_SELECT_REACTOR_TOKEN>::wait_for_multiple_events 5 (ACE_Select_Reactor_Handle_Set &dispatch_set, 6 ACE_Time_Value *max_wait_time) 7 { 8 ACE_TRACE ("ACE_Select_Reactor_T::wait_for_multiple_events"); 9 ACE_Time_Value timer_buf (0); 10 ACE_Time_Value *this_timeout = 0; 11 12 int number_of_active_handles = this->any_ready (dispatch_set); 13 14 // If there are any bits enabled in the <ready_set_> then we'll 15 // handle those first, otherwise we'll block in <select>. 16 17 if (number_of_active_handles == 0) 18 { 19 do 20 { 21 if (this->timer_queue_ == 0) 22 return 0; 23 24 this_timeout = 25 this->timer_queue_->calculate_timeout (max_wait_time, 26 &timer_buf); 27 #ifdef ACE_WIN32 28 // This arg is ignored on Windows and causes pointer 29 // truncation warnings on 64-bit compiles. 30 int const width = 0; 31 #else 32 int const width = this->handler_rep_.max_handlep1 (); 33 #endif /* ACE_WIN32 */ 34 35 dispatch_set.rd_mask_ = this->wait_set_.rd_mask_; 36 dispatch_set.wr_mask_ = this->wait_set_.wr_mask_; 37 dispatch_set.ex_mask_ = this->wait_set_.ex_mask_; 38 number_of_active_handles = ACE_OS::select (width, 39 dispatch_set.rd_mask_, 40 dispatch_set.wr_mask_, 41 dispatch_set.ex_mask_, 42 this_timeout); 43 } 44 while (number_of_active_handles == -1 && this->handle_error () > 0); 45 46 if (number_of_active_handles > 0) 47 { 48 #if !defined (ACE_WIN32) 49 // Resynchronize the fd_sets so their "max" is set properly. 50 dispatch_set.rd_mask_.sync (this->handler_rep_.max_handlep1 ()); 51 dispatch_set.wr_mask_.sync (this->handler_rep_.max_handlep1 ()); 52 dispatch_set.ex_mask_.sync (this->handler_rep_.max_handlep1 ()); 53 #endif /* ACE_WIN32 */ 54 } 55 else if (number_of_active_handles == -1) 56 { 57 // Normally, select() will reset the bits in dispatch_set 58 // so that only those filed descriptors that are ready will 59 // have bits set. However, when an error occurs, the bit 60 // set remains as it was when the select call was first made. 61 // Thus, we now have a dispatch_set that has every file 62 // descriptor that was originally waited for, which is not 63 // correct. We must clear all the bit sets because we 64 // have no idea if any of the file descriptors is ready. 65 // 66 // NOTE: We dont have a test case to reproduce this 67 // problem. But pleae dont ignore this and remove it off. 68 dispatch_set.rd_mask_.reset (); 69 dispatch_set.wr_mask_.reset (); 70 dispatch_set.ex_mask_.reset (); 71 } 72 } 73 74 // Return the number of events to dispatch. 75 return number_of_active_handles; 76 }

第35行熟悉的變量wait_set_和第38行函數(shù)select,到這里發(fā)現(xiàn),Reactor的I/O監(jiān)控,就是利用select函數(shù)監(jiān)控之前注冊(cè)進(jìn)去且ADD到wait_set_的handle,即I/O。

當(dāng)有I/O事件,即返回值的number_of_active_handles不為0時(shí),將進(jìn)行dispatch。

1 template <class ACE_SELECT_REACTOR_TOKEN> int 2 ACE_Select_Reactor_T<ACE_SELECT_REACTOR_TOKEN>::dispatch 3 (int active_handle_count, 4 ACE_Select_Reactor_Handle_Set &dispatch_set) 5 { 6 ACE_TRACE ("ACE_Select_Reactor_T::dispatch"); 7 8 int io_handlers_dispatched = 0; 9 int other_handlers_dispatched = 0; 10 int signal_occurred = 0; 11 // The following do/while loop keeps dispatching as long as there 12 // are still active handles. Note that the only way we should ever 13 // iterate more than once through this loop is if signals occur 14 // while we're dispatching other handlers. 15 16 do 17 { 18 // We expect that the loop will decrease the number of active 19 // handles in each iteration. If it does not, then something is 20 // inconsistent in the state of the Reactor and we should avoid 21 // the loop. Please read the comments on bug 2540 for more 22 // details. 23 int initial_handle_count = active_handle_count; 24 25 // Note that we keep track of changes to our state. If any of 26 // the dispatch_*() methods below return -1 it means that the 27 // <wait_set_> state has changed as the result of an 28 // <ACE_Event_Handler> being dispatched. This means that we 29 // need to bail out and rerun the select() loop since our 30 // existing notion of handles in <dispatch_set> may no longer be 31 // correct. 32 // 33 // In the beginning, our state starts out unchanged. After 34 // every iteration (i.e., due to signals), our state starts out 35 // unchanged again. 36 37 this->state_changed_ = false; 38 39 // Perform the Template Method for dispatching all the handlers. 40 41 // First check for interrupts. 42 if (active_handle_count == -1) 43 { 44 // Bail out -- we got here since <select> was interrupted. 45 if (ACE_Sig_Handler::sig_pending () != 0) 46 { 47 ACE_Sig_Handler::sig_pending (0); 48 49 // If any HANDLES in the <ready_set_> are activated as a 50 // result of signals they should be dispatched since 51 // they may be time critical... 52 active_handle_count = this->any_ready (dispatch_set); 53 54 // Record the fact that the Reactor has dispatched a 55 // handle_signal() method. We need this to return the 56 // appropriate count below. 57 signal_occurred = 1; 58 } 59 else 60 return -1; 61 } 62 63 // Handle timers early since they may have higher latency 64 // constraints than I/O handlers. Ideally, the order of 65 // dispatching should be a strategy... 66 else if (this->dispatch_timer_handlers (other_handlers_dispatched) == -1) 67 // State has changed or timer queue has failed, exit loop. 68 break; 69 70 // Check to see if there are no more I/O handles left to 71 // dispatch AFTER we've handled the timers... 72 else if (active_handle_count == 0) 73 return io_handlers_dispatched 74 + other_handlers_dispatched 75 + signal_occurred; 76 77 // Next dispatch the notification handlers (if there are any to 78 // dispatch). These are required to handle multi-threads that 79 // are trying to update the <Reactor>. 80 81 else if (this->dispatch_notification_handlers 82 (dispatch_set, 83 active_handle_count, 84 other_handlers_dispatched) == -1) 85 // State has changed or a serious failure has occured, so exit 86 // loop. 87 break; 88 89 // Finally, dispatch the I/O handlers. 90 else if (this->dispatch_io_handlers 91 (dispatch_set, 92 active_handle_count, 93 io_handlers_dispatched) == -1) 94 // State has changed, so exit loop. 95 break; 96 97 // if state changed, we need to re-eval active_handle_count, 98 // so we will not end with an endless loop 99 if (initial_handle_count == active_handle_count 100 || this->state_changed_) 101 { 102 active_handle_count = this->any_ready (dispatch_set); 103 } 104 } 105 while (active_handle_count > 0); 106 107 return io_handlers_dispatched + other_handlers_dispatched + signal_occurred; 108 }

這里一步步按順序進(jìn)行判斷分發(fā),進(jìn)入dispatch_notification_handlers,調(diào)用到Select_Reactor_Base.cpp的dispatch_notifications,到這里終于看到熟悉的函數(shù)。

1 // Handles pending threads (if any) that are waiting to unblock the 2 // Select_Reactor. 3 4 int 5 ACE_Select_Reactor_Notify::dispatch_notifications (int &number_of_active_handles, 6 ACE_Handle_Set &rd_mask) 7 { 8 ACE_TRACE ("ACE_Select_Reactor_Notify::dispatch_notifications"); 9 10 ACE_HANDLE const read_handle = 11 this->notification_pipe_.read_handle (); 12 13 if (read_handle != ACE_INVALID_HANDLE 14 && rd_mask.is_set (read_handle)) 15 { 16 --number_of_active_handles; 17 rd_mask.clr_bit (read_handle); 18 return this->handle_input (read_handle); 19 } 20 else 21 return 0; 22 }

第18行,調(diào)用了自己的handle_input,還不是最開(kāi)始外部重載的handle_input,查看最后這個(gè)函數(shù)。

1 int 2 ACE_Select_Reactor_Notify::handle_input (ACE_HANDLE handle) 3 { 4 ACE_TRACE ("ACE_Select_Reactor_Notify::handle_input"); 5 // Precondition: this->select_reactor_.token_.current_owner () == 6 // ACE_Thread::self (); 7 8 int number_dispatched = 0; 9 int result = 0; 10 ACE_Notification_Buffer buffer; 11 12 // If there is only one buffer in the pipe, this will loop and call 13 // read_notify_pipe() twice. The first time will read the buffer, and 14 // the second will read the fact that the pipe is empty. 15 while ((result = this->read_notify_pipe (handle, buffer)) > 0) 16 { 17 // Dispatch the buffer 18 // NOTE: We count only if we made any dispatches ie. upcalls. 19 if (this->dispatch_notify (buffer) > 0) 20 ++number_dispatched; 21 22 // Bail out if we've reached the <notify_threshold_>. Note that 23 // by default <notify_threshold_> is -1, so we'll loop until all 24 // the notifications in the pipe have been dispatched. 25 if (number_dispatched == this->max_notify_iterations_) 26 break; 27 } 28 29 // Reassign number_dispatched to -1 if things have gone seriously 30 // wrong. 31 if (result < 0) 32 number_dispatched = -1; 33 34 // Enqueue ourselves into the list of waiting threads. When we 35 // reacquire the token we'll be off and running again with ownership 36 // of the token. The postcondition of this call is that 37 // <select_reactor_.token_.current_owner> == <ACE_Thread::self>. 38 this->select_reactor_->renew (); 39 return number_dispatched; 40 }

第15行,取數(shù)據(jù),第19行dispatch_notify

1 int 2 ACE_Select_Reactor_Notify::dispatch_notify (ACE_Notification_Buffer &buffer) 3 { 4 int result = 0; 5 6 #if defined (ACE_HAS_REACTOR_NOTIFICATION_QUEUE) 7 // Dispatch one message from the notify queue, and put another in 8 // the pipe if one is available. Remember, the idea is to keep 9 // exactly one message in the pipe at a time. 10 11 bool more_messages_queued = false; 12 ACE_Notification_Buffer next; 13 14 result = notification_queue_.pop_next_notification(buffer, 15 more_messages_queued, 16 next); 17 18 if (result == 0 || result == -1) 19 { 20 return result; 21 } 22 23 if(more_messages_queued) 24 { 25 (void) ACE::send(this->notification_pipe_.write_handle(), 26 (char *)&next, sizeof(ACE_Notification_Buffer)); 27 } 28 #endif /* ACE_HAS_REACTOR_NOTIFICATION_QUEUE */ 29 30 // If eh == 0 then another thread is unblocking the 31 // <ACE_Select_Reactor> to update the <ACE_Select_Reactor>'s 32 // internal structures. Otherwise, we need to dispatch the 33 // appropriate handle_* method on the <ACE_Event_Handler> pointer 34 // we've been passed. 35 if (buffer.eh_ != 0) 36 { 37 ACE_Event_Handler *event_handler = buffer.eh_; 38 39 bool const requires_reference_counting = 40 event_handler->reference_counting_policy ().value () == 41 ACE_Event_Handler::Reference_Counting_Policy::ENABLED; 42 43 switch (buffer.mask_) 44 { 45 case ACE_Event_Handler::READ_MASK: 46 case ACE_Event_Handler::ACCEPT_MASK: 47 result = event_handler->handle_input (ACE_INVALID_HANDLE); 48 break; 49 case ACE_Event_Handler::WRITE_MASK: 50 result = event_handler->handle_output (ACE_INVALID_HANDLE); 51 break; 52 case ACE_Event_Handler::EXCEPT_MASK: 53 result = event_handler->handle_exception (ACE_INVALID_HANDLE); 54 break; 55 case ACE_Event_Handler::QOS_MASK: 56 result = event_handler->handle_qos (ACE_INVALID_HANDLE); 57 break; 58 case ACE_Event_Handler::GROUP_QOS_MASK: 59 result = event_handler->handle_group_qos (ACE_INVALID_HANDLE); 60 break; 61 default: 62 // Should we bail out if we get an invalid mask? 63 ACE_ERROR ((LM_ERROR, 64 ACE_TEXT ("invalid mask = %d\n"), 65 buffer.mask_)); 66 } 67 68 if (result == -1) 69 event_handler->handle_close (ACE_INVALID_HANDLE, 70 ACE_Event_Handler::EXCEPT_MASK); 71 72 if (requires_reference_counting) 73 { 74 event_handler->remove_reference (); 75 } 76 } 77 78 return 1; 79 }

到這里,終于看到調(diào)用到我們最開(kāi)始繼承ACE_Event_Handler重寫的那個(gè)回調(diào)handle_input()了。

?

至此,ACE_Reactor內(nèi)部源碼的執(zhí)行過(guò)程全部結(jié)束,其實(shí)ACE并沒(méi)有做非常特別的事,注冊(cè)利用一個(gè)容器進(jìn)行I/O和回調(diào)方法的綁定,I/O復(fù)用利用select,最后發(fā)生I/O事件找到對(duì)應(yīng)的event函數(shù)handle_input執(zhí)行。

怪不得網(wǎng)上有人抱怨ACE代碼臃腫了,這些我們關(guān)心“簡(jiǎn)單”過(guò)程的代碼就這么多的復(fù)雜用法和調(diào)用,更不用說(shuō)我們還沒(méi)用上的了,但是ACE提供的Reactor框架確實(shí)方便了我們使用,也提供了可靠的移植性和性能。

總結(jié)

以上是生活随笔為你收集整理的ACE - Reactor模式源码剖析及具体实现(大量源码慎入)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

国产精品丝袜 | 亚洲精品高清视频 | 欧美精品久久久久久久 | 日韩欧美国产精品 | 天天操天天艹 | 欧美日韩精品在线免费观看 | 欧美日韩视频免费 | 久久久久欠精品国产毛片国产毛生 | 狠狠久久婷婷 | 日韩高清在线一区 | 久久少妇免费视频 | 久久久这里有精品 | 在线观看免费观看在线91 | 国产一级视频免费看 | 精品久久在线 | 麻豆视频国产 | 99精品视频网站 | 精品国产福利在线 | 国产99精品在线观看 | 亚洲午夜小视频 | 色网站在线免费观看 | 青青草久草在线 | 蜜臀av性久久久久av蜜臀妖精 | 成年人黄色免费看 | 天天操天天操天天操天天操天天操天天操 | 精品美女在线视频 | 91在线最新| 国产精品mv在线观看 | 日韩av美女 | 在线www色| 国产一区视频在线 | 美女免费网站 | 最新国产精品久久精品 | 日本夜夜草视频网站 | 一区二区视频在线免费观看 | 国产高清视频免费在线观看 | 免费看av片网站 | 日韩小视频 | 成人一级在线观看 | 香蕉视频在线免费 | 成人久久18免费网站图片 | 亚洲成人资源在线 | 日日干日日 | 午夜久久久久久久久久影院 | 日本三级不卡 | 韩日av一区二区 | 久久免视频 | 国产久草在线 | 亚洲永久精品国产 | 国产久草在线观看 | www.国产在线观看 | 中文综合在线 | 日韩黄视频 | 欧美孕妇与黑人孕交 | 精品一区二区三区香蕉蜜桃 | 国产人成免费视频 | 福利一区二区 | 欧美成人xxx| 少妇av片| 91黄色视屏| 丝袜美腿在线视频 | 国产91国语对白在线 | 久久精品视频在线播放 | 亚洲精品视频国产 | 欧美在线观看视频免费 | 综合伊人久久 | 国产亚洲精品美女久久 | 久久国产精品一国产精品 | 日韩小视频 | 91免费日韩 | 欧美在线观看小视频 | 天天综合网久久 | 午夜丁香网| 国产精品午夜免费福利视频 | 亚洲国产综合在线 | 成人资源站 | 亚洲欧美日韩一区二区三区在线观看 | 看片的网址 | 丁香影院在线 | 亚洲黄色激情小说 | 久久久久久中文字幕 | 久久精品国产精品亚洲精品 | 日日干夜夜草 | 91福利视频网站 | 久久免费视频这里只有精品 | 欧美日韩综合在线观看 | 国产精品久久久久久婷婷天堂 | av网站在线免费观看 | 欧美日韩99 | 精品嫩模福利一区二区蜜臀 | 国产成人在线精品 | 91激情小视频 | 色.www| 免费久久99精品国产 | 国产成人中文字幕 | 国产精品久久久久婷婷二区次 | 日本精品一区二区三区在线播放视频 | 精久久久久 | 91免费视频网站在线观看 | 中字幕视频在线永久在线观看免费 | 狠狠色伊人亚洲综合网站色 | 国产最新在线 | 在线播放日韩 | av色网站 | 东方av在线免费观看 | 高清精品久久 | av资源免费看 | 成人三级网址 | 国产真实在线 | 国产精品视频一二三 | 99r在线观看 | 中文久草 | www.xxx.性狂虐 | 国产精品永久免费 | 精品96久久久久久中文字幕无 | 91精品视频在线看 | 伊人网av| 久久99精品国产一区二区三区 | 网站你懂的 | 操操操日日 | 国产小视频福利在线 | 9ⅰ精品久久久久久久久中文字幕 | 国产精品精品久久久久久 | 国产精品视频地址 | 久久亚洲精品国产亚洲老地址 | 国产高清日韩 | 91少妇精拍在线播放 | 欧美日韩在线精品一区二区 | 国产精品一区二区在线观看 | 精品色综合 | 久久久精品福利视频 | 在线播放国产一区二区三区 | 3d黄动漫免费看 | 国产小视频在线看 | 在线播放日韩 | 精品国产精品国产偷麻豆 | 6699私人影院 | 久久久亚洲网站 | 国产精品美女久久久久久免费 | 九色自拍视频 | 国内成人精品2018免费看 | 成人9ⅰ免费影视网站 | 97色婷婷成人综合在线观看 | 国产一级久久久 | 国产最新视频在线 | 日韩欧美一区二区三区视频 | 黄色特级片 | 午夜12点| 国产色视频123区 | 五月婷婷综合在线视频 | 99视频网站 | 日韩h在线观看 | 又色又爽又黄 | 日韩a在线观看 | 久久国产片 | 成人av av在线 | 视频一区二区免费 | 久久精品一区二区三区国产主播 | 狠狠色丁香婷婷综合橹88 | 在线免费中文字幕 | 精品国产成人在线 | 午夜视频在线观看欧美 | 国产精品久久久久久久久蜜臀 | 亚洲国产成人精品电影在线观看 | av一级久久 | 国产精品免费不卡 | 色综合 久久精品 | 国产麻豆精品95视频 | 亚洲国产日本 | www看片网站 | 黄色av一级片 | 欧美视频网址 | 91精品久久久久久综合乱菊 | 天天天干天天射天天天操 | 亚洲va天堂va欧美ⅴa在线 | 久久精品高清视频 | 91激情 | 国产视频九色蝌蚪 | 国产免费观看高清完整版 | 九九久久电影 | 久久成人国产精品 | 成人av高清| 日本中文乱码卡一卡二新区 | 人人干人人干人人干 | 色综合久久精品 | 色橹橹欧美在线观看视频高清 | 在线视频 成人 | 久久首页 | 伊人永久 | 人人狠狠综合久久亚洲婷 | 三上悠亚一区二区在线观看 | 午夜免费在线观看 | 一色屋精品视频在线观看 | 99精品免费久久久久久久久日本 | 欧美在线1区 | 亚洲精品成人网 | 婷婷色伊人 | 丝袜制服天堂 | 久久精品在线视频 | 免费观看的av | 自拍超碰在线 | 国产成人一二三 | 国产亚洲精品v | 一区二区 精品 | 国产精品免费视频一区二区 | 中文字幕一区二区三区在线播放 | 伊人天天狠天天添日日拍 | 久久av在线播放 | 亚洲国产精品va在线看黑人 | 免费午夜av | 成人h视频 | 99热99re6国产在线播放 | 免费日韩一区二区 | 日韩免费一区二区在线观看 | 日日干夜夜操视频 | 精品99久久久久久 | 色婷婷久久一区二区 | 精品久久久久一区二区国产 | 亚洲免费黄色 | 伊人午夜 | 97视频久久久| 中文字幕在线观看免费观看 | 在线看av的网址 | 国产精品久久久久一区二区国产 | 香蕉视频在线免费 | 黄色小说视频网站 | 亚洲a成人v | 国产小视频精品 | 亚洲视频在线观看免费 | 欧美精品一二三 | 美腿丝袜av | 免费看的黄网站 | 99精品观看 | 久久久久久久久久久网站 | 欧美精品免费在线观看 | 精品 一区 在线 | www.五月婷婷.com | 成年人免费电影 | 免费视频久久久 | 中文av日韩 | 国产尤物视频在线 | 91成人免费在线视频 | 国产成人精品一区二区三区在线 | 亚洲精品国产品国语在线 | 狠狠色丁香久久婷婷综 | 91福利视频免费 | 国产丝袜在线 | 亚州av网站| 久久毛片高清国产 | 国产精品涩涩屋www在线观看 | 91丨九色丨国产丨porny精品 | 色婷婷精品大在线视频 | 美女免费电影 | 国产成人精品久久久久蜜臀 | 亚洲综合爱 | 日本久久久亚洲精品 | 国产小视频在线看 | 99久久久久久久久 | 久久精品视频在线免费观看 | 国产小视频在线免费观看视频 | 精品日韩在线一区 | 绯色av一区| 国语对白少妇爽91 | 一区二区三区观看 | 五月天丁香综合 | 欧美日韩性生活 | 九九热在线播放 | 国产高清视频在线 | 国产日韩欧美中文 | 亚洲欧洲精品一区二区精品久久久 | 午夜av免费在线观看 | 91视频 - x99av | 99久久精品费精品 | 我要色综合天天 | 国产精品乱看 | 在线高清一区 | 欧美黄色特级片 | 97精品国产97久久久久久免费 | 99这里只有 | 日韩精品五月天 | 色婷婷久久 | 成人毛片网 | 干亚洲少妇 | 精品极品在线 | 久久草视频 | 91最新网址在线观看 | 六月婷婷久香在线视频 | 亚洲欧美国产视频 | 国产黄在线免费观看 | 91喷水| 国产 日韩 欧美 在线 | 精品国产伦一区二区三区观看体验 | 伊色综合久久之综合久久 | 精品在线观看一区二区三区 | 久久不见久久见免费影院 | 日本精品视频免费观看 | 中文字幕在线专区 | 一二三区高清 | av+在线播放在线播放 | www.xxx.性狂虐 | 久久国产91 | 特级西西人体444是什么意思 | 久久试看 | 美女黄濒 | 国产精品激情偷乱一区二区∴ | 亚洲精品tv | 人人狠狠综合久久亚洲婷 | 337p日本欧洲亚洲大胆裸体艺术 | 久久精品网站免费观看 | 欧美一区影院 | 黄色成年| 成人av电影免费在线播放 | 国产精品18毛片一区二区 | 91在线精品播放 | 丁香视频全集免费观看 | 正在播放日韩 | 久草精品电影 | 国产精品一区一区三区 | 久久第四色 | 久久爱综合 | 在线亚洲高清视频 | 免费又黄又爽视频 | 人人爽久久久噜噜噜电影 | 亚洲永久精品国产 | 成人免费网站视频 | 国产成人精品亚洲精品 | 久久激情视频免费观看 | 一区二区电影在线观看 | 欧美黄网站 | 美女视频久久 | 在线99热| 亚州国产精品视频 | 男女日麻批 | 国产精品6 | 爱干视频| 永久免费精品视频 | 久久视频中文字幕 | 欧美一级专区免费大片 | 精品免费视频123区 午夜久久成人 | 我要看黄色一级片 | 欧美精品一区在线 | 色干干 | 91麻豆文化传媒在线观看 | 精品国产乱码久久 | a天堂最新版中文在线地址 久久99久久精品国产 | 精品亚洲视频在线 | 亚洲一级免费电影 | 中文字幕精品视频 | 97视频在线观看视频免费视频 | a视频在线 | 国产91精品欧美 | 免费精品在线观看 | 国产精品久久久久久久久久久久冷 | 国产精品久久久久久模特 | 激情视频国产 | 中文字幕999| 97视频网站 | 国产黄色资源 | 成人网在线免费视频 | 国产生活一级片 | 免费观看一级特黄欧美大片 | 精品美女久久久久久免费 | 国产亚洲精品久久 | av免费观看网址 | 国产无限资源在线观看 | 97免费在线观看视频 | 国产一区二区不卡在线 | 精品一区三区 | 91在线免费播放 | 一级片视频在线 | 69国产成人综合久久精品欧美 | 特黄特黄的视频 | 国产69精品久久99不卡的观看体验 | av青草 | 国内精品久久久久影院一蜜桃 | 天天色天天色天天色 | 天天搞天天干 | 亚洲成人av电影 | 久久天堂网站 | 欧美午夜精品久久久久久浪潮 | 深爱婷婷 | 日韩视频一区二区三区在线播放免费观看 | 99热亚洲精品 | 日韩精品免费一区二区 | 国产精品99久久免费黑人 | 欧美国产三区 | 99久久精品免费看国产四区 | 99久久夜色精品国产亚洲96 | 99国产精品免费网站 | 久草视频资源 | 国产免费视频在线 | 久久久久免费精品国产小说色大师 | 精品毛片一区二区免费看 | 久久久久久久久久国产精品 | 成人wwwxxx视频 | 免费污片 | 在线天堂中文www视软件 | 精品国产a | 岛国av在线 | 欧美a级片网站 | 亚洲视频观看 | 天天爽综合网 | 人人爽人人舔 | 日韩欧美一区视频 | 美女视频网站久久 | 操操操夜夜操 | 免费在线观看不卡av | 国产精品久久久一区二区三区网站 | 久久国产精品免费一区二区三区 | 久久综合久久综合久久综合 | 成人av手机在线 | 9999在线观看| 精品国产一区二区三区av性色 | 九九热av | 99久热在线精品视频 | 中文字幕在线网址 | 欧美日韩视频在线观看免费 | 五月导航 | 99久久久国产精品免费观看 | 中文字幕在线电影 | 天天操比 | 超碰官网| 天天射天天射天天射 | 日韩久久精品一区二区 | 四虎在线免费视频 | 国产盗摄精品一区二区 | 国产乱视频| av电影免费在线看 | 久久精品中文字幕一区二区三区 | 国产一区二区三区高清播放 | 人人干免费 | 精品国产99| 二区视频在线 | 天天色天天干天天色 | 国产日韩视频在线播放 | 亚洲一区二区黄色 | 日韩高清在线一区二区三区 | 视频在线一区二区三区 | 亚洲精品免费在线播放 | 在线观看91视频 | 在线观看国产福利片 | 一级黄色片在线观看 | 在线电影av | 久久精品这里热有精品 | 欧美日韩在线看 | 欧美三级免费 | 日本特黄一级 | 色wwww| 毛片视频电影 | 国产精品手机在线播放 | 91综合视频在线观看 | 免费a网站 | 中文字幕91视频 | 亚洲国产欧洲综合997久久, | 在线观看黄a | 色噜噜在线观看视频 | 天天操天天色综合 | 欧美性做爰猛烈叫床潮 | 国产精品 中文字幕 亚洲 欧美 | 激情www | 久久国产精品视频免费看 | 蜜臀av性久久久久蜜臀aⅴ涩爱 | 欧美激情综合网 | 日韩小视频网站 | 欧美日本国产在线观看 | 欧美成人精品欧美一级乱黄 | 少妇bbw搡bbbb搡bbb | 亚洲三级黄 | 九九九九热精品免费视频点播观看 | 欧美综合在线视频 | 亚洲成人午夜在线 | 天天综合天天做 | 97超视频在线观看 | 久久久在线 | 亚洲激情精品 | 久久免费黄色大片 | 天天天天天天干 | 国产成人一区在线 | 日本久久久久 | 国产成人精品午夜在线播放 | 日韩黄视频 | 五月天久久精品 | 精品一区二区在线看 | 国产精品中文在线 | 天天干天天碰 | 亚洲国产成人高清精品 | av中文在线影视 | 久久99国产精品久久99 | 最近中文字幕高清字幕免费mv | 久久精品婷婷 | 国产成人久久av | 97成人超碰| 美女视频a美女大全免费下载蜜臀 | 免费观看一区二区 | 久久午夜免费观看 | 男女日麻批 | 激情欧美一区二区三区免费看 | 国产精品久久久久久久毛片 | 成人午夜黄色影院 | 日韩免费电影一区二区三区 | 96国产精品视频 | 久久久久久国产一区二区三区 | 在线精品观看 | 亚洲精品毛片一级91精品 | av手机版 | 久草国产精品 | 亚洲一区不卡视频 | 狠狠色噜噜狠狠狠狠2021天天 | 91人人网| 免费精品在线视频 | 久久精品美女视频网站 | 粉嫩一区二区三区粉嫩91 | 伊人狠狠操 | 九九免费观看全部免费视频 | 99国产在线观看 | 国产一级片一区二区三区 | 日韩精品播放 | 久久久一本精品99久久精品 | 国产高清av免费在线观看 | 国产综合片 | 久草精品在线观看 | 久草新在线 | 亚洲美女免费精品视频在线观看 | 亚洲精品乱码久久久久v最新版 | 五月天综合网站 | 一区二区三区高清在线 | 91麻豆精品久久久久久 | 国产一区91 | 久久激情视频 久久 | av先锋中文字幕 | 婷婷久久五月天 | 激情开心网站 | 国产精品区在线观看 | 亚洲视频www | 国产理论在线 | 好看的国产精品视频 | 国产精品美女久久 | 99久久超碰中文字幕伊人 | 丁香婷婷综合色啪 | 日韩精品一区二区三区在线视频 | 久久天天躁夜夜躁狠狠躁2022 | 久久综合狠狠综合久久综合88 | 99久久这里有精品 | 玖玖视频| 精品久久网站 | 久草热久草视频 | 最近中文字幕高清字幕在线视频 | 天天干天天操天天干 | 欧美一区二区免费在线观看 | 伊人午夜视频 | 91麻豆产精品久久久久久 | 亚洲国产欧美在线看片xxoo | 色网站在线看 | 久久久亚洲麻豆日韩精品一区三区 | 永久免费视频国产 | 成人欧美一区二区三区黑人麻豆 | 少妇bbw揉bbb欧美 | 五月婷婷六月丁香激情 | 丁香六月婷婷开心婷婷网 | 欧美9999| 亚洲欧洲日韩在线观看 | 黄色免费观看网址 | 天堂av一区二区 | а中文在线天堂 | 草久久久 | 国产91欧美 | 97人人添人澡人人爽超碰动图 | 中文亚洲欧美日韩 | 国产精品99蜜臀久久不卡二区 | 狠狠色丁香九九婷婷综合五月 | 日韩区欧美久久久无人区 | 国产一区私人高清影院 | 99久久国产免费,99久久国产免费大片 | 久久精品中文字幕 | 久久黄色片子 | 97视频免费观看2区 亚洲视屏 | 欧美日韩在线观看一区二区 | 久久高清免费观看 | 六月激情 | 日韩激情小视频 | 91视频 - v11av | 国产精品久久久亚洲 | 天天操天天干天天操天天干 | 手机av在线免费观看 | 视频国产在线观看18 | 国产正在播放 | 97香蕉超级碰碰久久免费软件 | 91在线资源| 免费看黄电影 | 久久久久久久久久久黄色 | 日韩素人在线观看 | 五月天天av | 日韩av在线看| 91精品国产自产在线观看 | 亚洲精品女 | 成人一区二区三区在线 | 亚洲在线免费视频 | 98福利在线 | 欧美另类成人 | 蜜臀久久99精品久久久久久网站 | 91传媒免费观看 | 亚洲国产中文字幕 | 日韩视频在线一区 | 日b视频在线观看网址 | 天天色天天干天天 | 国产成人性色生活片 | 中文字幕成人av | 91九色蝌蚪国产 | 国产亚洲成av片在线观看 | 亚洲精品美女在线观看播放 | 在线国产福利 | 国产日韩在线看 | 人人爽久久涩噜噜噜网站 | 中文字幕乱码视频 | 97精品免费视频 | 日韩影视精品 | 国产视频一区二区三区在线 | 夜夜躁狠狠躁 | 国际精品久久久 | 亚洲精品中文在线观看 | 中国精品少妇 | 亚洲国产免费网站 | 婷婷丁香激情 | 九九热只有这里有精品 | 亚洲伊人成综合网 | 在线观看91视频 | 亚洲每日更新 | 久久九九久久九九 | 日韩av免费在线看 | 高清一区二区三区 | 在线香蕉视频 | 国产一二区免费视频 | 国产精品自在欧美一区 | 久草在线资源视频 | 亚洲免费观看视频 | 久久精品这里热有精品 | 亚洲片在线 | 日韩在线精品视频 | 97超级碰碰碰视频在线观看 | 99热精品免费观看 | 成人精品福利 | 国产一级精品绿帽视频 | 在线三级中文 | 成人午夜片av在线看 | 国产精品igao视频网入口 | 国产成人精品一二三区 | 色婷婷福利 | 香蕉影视 | 国产一二三区av | 国产理论片在线观看 | 亚洲精品乱码久久久久久蜜桃动漫 | 国产精品片 | 在线观看精品一区 | 久草在线91 | 天天在线视频色 | 国产一级片免费播放 | 久久精品网址 | 亚洲成a人片综合在线 | 天堂在线视频免费观看 | 日韩中字在线观看 | 国产精品专区一 | www狠狠操 | 久久久国产精品成人免费 | 国产黄色在线看 | 日韩视| 国产成人av网| 天堂在线v| 福利片视频区 | 欧美黄污视频 | 国产精品九九视频 | 男女视频久久久 | 在线日韩视频 | av大片免费| 成人免费视频视频在线观看 免费 | 国产一区成人在线 | 欧美成年人在线视频 | 中文字幕在线资源 | 久久免费看av | 亚洲国产精品久久久 | 久久久久国产成人免费精品免费 | 夜夜视频欧洲 | 成年免费在线视频 | 91av影视 | 亚洲成a人片77777kkkk1在线观看 | 偷拍区另类综合在线 | 国产精品理论在线观看 | 色综合久久88色综合天天人守婷 | 五月综合在线观看 | 国内精自线一二区永久 | 国产 欧美 在线 | 伊人伊成久久人综合网小说 | 婷婷六月激情 | 99热精品国产 | 久草在线综合网 | 天天射一射 | 四虎免费在线观看视频 | 成人黄色小说视频 | 精品亚洲网 | 日本精品一二区 | 91麻豆精品国产自产 | 激情五月六月婷婷 | 日韩欧三级 | 亚洲一区尤物 | 特级aaa毛片| 91av小视频| 日韩美女av在线 | 午夜精品久久久久久久久久久 | 99久久精品国产欧美主题曲 | 天天爱天天操 | 天天综合导航 | 成人在线超碰 | 怡红院av| 在线观看日韩精品 | 欧美a视频在线观看 | 中文字幕日韩有码 | 日韩高清免费在线 | 亚洲一二视频 | 五月天精品视频 | 在线日韩视频 | 国产伦精品一区二区三区四区视频 | 国产xxxx做受性欧美88 | 国产v在线播放 | 一区二区精品国产 | 五月婷婷综合激情网 | 亚洲狠狠丁香婷婷综合久久久 | 久久精品国产一区二区三区 | 中文乱幕日产无线码1区 | 国产一级片免费视频 | 最近中文字幕视频完整版 | 夜夜操狠狠干 | 天天激情站 | 成人h电影| 婷婷精品国产一区二区三区日韩 | 免费av片在线 | 在线国产精品视频 | 人人射av | 色婷婷久久一区二区 | 久久99精品国产99久久6尤 | 国产在线黄 | 成人黄色av免费在线观看 | 激情综合交 | 中文字幕在线看视频国产 | 亚洲精品美女久久17c | 人人插人人射 | 国产精品区免费视频 | 国产在线视频导航 | 国产一级高清 | 免费观看91视频大全 | 久久不卡日韩美女 | 亚洲三级在线免费观看 | 欧美精品在线观看免费 | 国产精品高清av | 日韩电影在线看 | 免费在线成人av电影 | 99精品美女 | 激情av五月婷婷 | 不卡av免费在线观看 | 手机在线日韩视频 | 日韩字幕 | av免费在线观看1 | 国产精品久久久久久一区二区 | 国产综合在线观看视频 | 国产成人一级 | 天天干天天色2020 | 狠狠狠狠狠狠天天爱 | 国产精品伦一区二区三区视频 | 精品国产区在线 | 久久久久一区二区三区 | 91爱爱网址 | 91精品国产91久久久久 | 中文在线字幕免 | 亚洲精品综合在线 | 欧美aa在线 | 日韩免费电影 | 国产无吗一区二区三区在线欢 | 中文字幕高清免费日韩视频在线 | 97天天综合网 | 91看片看淫黄大片 | 99久久日韩精品视频免费在线观看 | 免费av在线网 | 在线91播放 | 国产一区二区免费 | 天天艹 | 久草电影在线观看 | 超碰人人射| 精品一区二区在线播放 | 在线免费观看一区二区三区 | 精产嫩模国品一二三区 | 在线99热 | 日本久久免费电影 | 亚洲一级二级三级 | 久久久国产精品视频 | 国产精品99精品久久免费 | 亚洲专区免费观看 | 免费69视频| 天天干天天操天天 | 色中文字幕在线观看 | 99色资源 | 欧美激情精品久久久久 | 精品在线视频播放 | 狠狠操综合 | 成人不用播放器 | 久99久中文字幕在线 | 国产又粗又猛又色 | 麻豆国产视频 | 国产午夜精品久久 | 亚洲婷婷免费 | 精品99视频 | 国产一级免费视频 | av福利电影 | www.成人久久 | 久久精品久久99精品久久 | 女人18毛片90分钟 | 欧美激情第一区 | 香蕉视频网址 | 人人搞人人爽 | 色婷婷狠狠五月综合天色拍 | 亚洲一区日韩在线 | 亚洲一区二区三区毛片 | 久久久久久久久久免费视频 | 久久久精品免费看 | 国产私拍在线 | av资源免费看 | 国产特级毛片aaaaaa毛片 | 麻豆va一区二区三区久久浪 | 亚洲精品在线视频网站 | 亚洲国产97在线精品一区 | 69视频在线| 五月天六月色 | 久久久久久久久艹 | 久久久国产精品一区二区中文 | 国产精品久久久久久久久费观看 | 久久精品综合视频 | 福利视频午夜 | 久久久国内精品 | 五月综合在线观看 | 国产精品成人一区二区 | 天天综合网 天天综合色 | 久久人人干 | 中文字幕欲求不满 | 狠狠狠狠狠狠天天爱 | 国产精品久久久久久久免费观看 | 香蕉视频91 | 1024在线看片 | 国产午夜一区二区 | 精品国产免费一区二区三区五区 | 午夜精品久久久久久久99 | 久久久久网址 | 成人av资源网 | 久久综合欧美精品亚洲一区 | 狠狠躁日日躁夜夜躁av | 蜜桃视频在线视频 | 久久精品视频网 | 在线国产日韩 | 久久久免费观看视频 | av在线一级 | 欧美精品日韩 | 免费在线激情电影 | 久久欧美在线电影 | 天天射天天操天天 | 日韩动漫免费观看高清完整版在线观看 | 91尤物在线播放 | 午夜国产一区二区三区四区 | 亚洲视频六区 | 久久婷婷一区 | 久久电影国产免费久久电影 | 精品久久久久久久久久久久久久久久久久 | 在线观看日韩视频 | 中文资源在线官网 | 美女视频黄,久久 | 婷婷激情影院 | 亚洲第一伊人 | 亚洲国产美女久久久久 | 国产精品久久久久9999吃药 | 69国产成人综合久久精品欧美 | 日本精品视频在线观看 | 91精品爽啪蜜夜国产在线播放 | 亚洲精品在线一区二区三区 | 91激情在线视频 | 五月婷婷激情六月 | 国产成人精品一区二区三区免费 | 中文字幕在线一区观看 | 色橹橹欧美在线观看视频高清 | 国产精品久久久久久久久蜜臀 | 日韩区欠美精品av视频 | 亚洲精品视频免费在线 | 亚洲欧美日韩精品久久久 | 国产黄色片久久久 | 狠狠躁天天躁 | h动漫中文字幕 | 免费一级特黄毛大片 | 深爱激情五月综合 | 国产在线传媒 | 国产美女久久 | 亚洲影音先锋 | 国产 精品 资源 | 亚洲精品一区二区18漫画 | 偷拍视频一区 | 日韩av在线看 | 91精品久久久久久综合五月天 | 国产成人亚洲在线观看 | 深爱激情站| 天天爽天天爽夜夜爽 | www.国产在线视频 | 日韩网站免费观看 | 91精品国产99久久久久 | 国产xx视频| 久草在线 | 91色在线观看视频 | 在线观看激情av | 成年人网站免费在线观看 | 超碰在线亚洲 | 在线免费高清一区二区三区 | 五月天高清欧美mv | 丁香亚洲 | 国产一级做a爱片久久毛片a | 日韩av黄 | 久久久久久美女 | 欧美日韩国产三级 | 亚洲精品xxxx| 午夜视频免费播放 | 天天狠狠干 | 蜜桃视频在线观看一区 | 色天天| 日韩成人精品在线观看 | 天天天综合网 | 精品久久久久久一区二区里番 | 亚洲高清av | 在线观看国产亚洲 | 九九有精品 | 精品99视频 | 亚洲精品视频偷拍 | 久久99国产精品久久99 | 亚洲成人精品在线观看 | 国产视频 亚洲精品 | 超碰人人超 | 久草在线中文888 | 六月丁香激情网 | 在线观看中文字幕视频 | 伊人婷婷网 | 91麻豆福利| 亚洲成熟女人毛片在线 | 久久特级毛片 | 国产精品手机看片 | av成年人电影 | 国产99久久久欧美黑人 | 色欧美成人精品a∨在线观看 | 国产精品99久久久久久武松影视 | 超碰在线人人爱 | 国产色妞影院wwwxxx | 成人国产电影在线观看 | 天天亚洲 | 亚洲精品乱码久久久久久蜜桃91 | 九九九九热精品免费视频点播观看 | 国产高清av免费在线观看 | 久久资源总站 | 免费观看一区二区三区视频 | 亚洲天堂网站视频 | 婷婷在线色 | 国产涩图| 91成人免费在线视频 | 91精品国产91久久久久福利 | 日日爽天天 | 国产不卡av在线 | 综合网婷婷 | 91看片在线观看 | 91日韩免费 | 精品国产资源 | 久久系列 | 看片网站黄色 | 国产精品丝袜久久久久久久不卡 | 国产在线播放观看 | 亚洲电影一区二区 | 久久精品视频免费 | 99在线视频观看 | 成人av亚洲 | 婷婷丁香在线 | 成人毛片一区 | 国产 色 | 成人一区二区在线 | 日韩av电影中文字幕 | 欧美日韩啪啪 | 日韩视频免费在线 | 99精品免费在线观看 | 91av视频在线免费观看 | 中文乱幕日产无线码1区 | 免费人成在线观看 | 久久经典视频 | 91成人免费看 | 久久6精品 | 亚洲欧美日韩精品一区二区 | 99亚洲精品在线 | 国产1区在线观看 | www.看片网站 | 91热这里只有精品 | 婷婷国产v亚洲v欧美久久 | 成人久久毛片 | 五月天婷婷视频 |