• <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>
            隨筆-380  評論-37  文章-0  trackbacks-0
            ASIO的異步方式
            異步方式 和同步方式不同,從來不花時(shí)間去等那些龜速的IO操作,只是向系統(tǒng)說一聲要做什么,然后就可以做其它事去了。
            如果系統(tǒng)完成了操作, 系統(tǒng)就會通過我之前給它的回調(diào)對象來通知我。
            在ASIO庫中,異步方式的函數(shù)或方法名稱前面都有“async_” 前綴,函數(shù)參數(shù)里會要求放一個回調(diào)函數(shù)(或仿函數(shù))。
            異步操作執(zhí)行 后不管有沒有完成都會立即返回,這時(shí)可以做一些其它事,直到回調(diào)函數(shù)(或仿函數(shù))被調(diào)用,說明異步操作已經(jīng)完成。
            在ASIO中很多回調(diào)函數(shù)都只接受一個boost::system::error_code參數(shù),在實(shí)際使用時(shí)肯定是不夠的,所以一般 使用仿函數(shù)攜帶一堆相關(guān)數(shù)據(jù)作為回調(diào),或者使用boost::bind來綁定一堆數(shù)據(jù)。
            另外要注意的是,只有io_service類的run()方法運(yùn)行之后回調(diào)對象才會被調(diào)用,否則即使系統(tǒng)已經(jīng)完成了異步操作也不會有任 務(wù)動作。
            好了,就介紹到這里,下面是我?guī)淼漠惒椒绞絋CP Helloworld服務(wù)器端:
            1.#include
            2.#include
            3.#include
            4.#include
            5.#include
            6.
            7.using namespace boost::asio;
            8.using boost::system::error_code;
            9.using ip::tcp;
            10.
            11.struct CHelloWorld_Service{
            12. CHelloWorld_Service(io_service &iosev)
            13. :m_iosev(iosev),m_acceptor(iosev, tcp::endpoint(tcp::v4(), 1000))
            14. {
            15. }
            16.
            17. void start()
            18. {
            19. // 開始等待連接(非阻塞)
            20. boost::shared_ptr psocket(new tcp::socket(m_iosev));
            21. // 觸發(fā)的事件只有error_code參數(shù),所以用boost::bind把socket綁定進(jìn)去
            22. m_acceptor.async_accept(*psocket,
            23. boost::bind(&CHelloWorld_Service::accept_handler, this, psocket, _1)
            24. );
            25. }
            26.
            ?27. // 有客戶端連接時(shí)accept_handler觸發(fā)
            28. void accept_handler(boost::shared_ptr psocket, error_code ec)
            29. {
            30. if(ec) return;
            31. // 繼續(xù)等待連接
            32. start();
            ?33. // 顯示遠(yuǎn)程IP
            ?34. std::cout << psocket->remote_endpoint().address() << std::endl;
            35. // 發(fā)送信息(非阻塞)
            36. boost::shared_ptr pstr(new std::string("hello async world!"));
            37. psocket->async_write_some(buffer(*pstr),
            38. boost::bind(&CHelloWorld_Service::write_handler, this, pstr, _1, _2)
            39. );
            ?40. }
            ?41.
            42. // 異步寫操作完成后write_handler觸發(fā)
            43. void write_handler(boost::shared_ptr pstr,
            44. error_code ec, size_t bytes_transferred)
            45. {
            46. if(ec)
            47. std::cout<< "發(fā)送失敗!" << std::endl;
            48. else
            49. std::cout<< *pstr << " 已發(fā)送" << std::endl;
            50. }
            51.
            52.private:
            53. io_service &m_iosev;
            54. ip::tcp::acceptor m_acceptor;
            55.};
            56.
            57.int main(int argc, char* argv[])
            58.{
            59. io_service iosev;
            60. CHelloWorld_Service sev(iosev);
            ?61. // 開始等待連接
            62. sev.start();
            63. iosev.run();
            64.
            65. return 0;
            66.}
            在這個例子中,首先調(diào)用sev.start()開 始接受客戶端連接。
            由于async_accept調(diào) 用后立即返回,start()方 法 也就馬上完成了。
            sev.start()在 瞬間返回后iosev.run()開 始執(zhí)行,iosev.run()方法是一個循環(huán),負(fù)責(zé)分發(fā)異步回調(diào)事件,
            只 有所有異步操作全部完成才會返回。
            這里有個問題,就是要保證start()方法中m_acceptor.async_accept操 作所用的tcp::socket對象在整個異步操作期間保持有效
            (不 然系統(tǒng)底層異步操作了一半突然發(fā)現(xiàn)tcp::socket沒了,不是拿人家開涮嘛-_-!!!),
            而且客戶端連接進(jìn)來后這個tcp::socket對象還 有用呢。
            這里的解決辦法是使用一個帶計(jì)數(shù)的智能指針boost::shared_ptr,并把這個指針作為參數(shù)綁定到回調(diào)函數(shù)上。
            一旦有客戶連接,我們在start()里給的回調(diào)函數(shù)accept_handler就會被 調(diào)用,
            首先調(diào)用start()繼續(xù)異步等待其 它客戶端的連接,然后使用綁定進(jìn)來的tcp::socket對象與當(dāng)前客戶端通信。
            發(fā)送數(shù)據(jù)也使用了異步方式(async_write_some), 同樣要保證在整個異步發(fā)送期間緩沖區(qū)的有效性,
            所以也用boost::bind綁定了boost::shared_ptr
            對于客戶端也一樣,在connect和read_some方法前加一個async_前綴,然后加入回調(diào)即可,大家自己練習(xí)寫一寫。
            posted on 2009-08-19 00:36 小王 閱讀(1019) 評論(0)  編輯 收藏 引用 所屬分類: 網(wǎng)絡(luò)通訊
            久久天天躁狠狠躁夜夜2020一| 精品国产乱码久久久久久呢| 久久精品国产清自在天天线| 久久久精品无码专区不卡| 国产真实乱对白精彩久久| 日本精品久久久中文字幕| 中文字幕精品无码久久久久久3D日动漫 | 国产精品乱码久久久久久软件| 色播久久人人爽人人爽人人片AV | 国产精自产拍久久久久久蜜| 国产精品伊人久久伊人电影| 最新久久免费视频| 国产精品久久久久久| 中文字幕久久精品 | 久久亚洲色一区二区三区| 久久精品国产男包| 久久国产影院| 一本色道久久88加勒比—综合| 成人综合久久精品色婷婷| 久久精品无码免费不卡| 精品国产福利久久久| 久久国产亚洲精品无码| 久久精品国产亚洲AV不卡| 一本一本久久A久久综合精品 | 亚洲色大成网站WWW久久九九| 国产一区二区三精品久久久无广告| 漂亮人妻被黑人久久精品| 久久久久亚洲精品无码网址| 伊人久久一区二区三区无码| 亚洲а∨天堂久久精品9966| 99热成人精品免费久久| 国内精品久久久久久久涩爱| 手机看片久久高清国产日韩| 天堂无码久久综合东京热| 日韩精品久久久久久久电影| 久久99亚洲网美利坚合众国| 91精品国产9l久久久久| 久久国产成人午夜aⅴ影院 | 久久久久人妻一区二区三区vr| 久久久久99精品成人片欧美| 香蕉aa三级久久毛片|