• <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>

            ACG狂人

            其實(shí)我更愛(ài)姐汁...

            ASIO攻破!?。?/a>

            花了足足3天時(shí)間,外加1天心情休整,終于在第5天編寫(xiě)出了一個(gè)能運(yùn)行的基于asio和thread_group的框架,差點(diǎn)沒(méi)氣暈過(guò)去,把源碼都看懂了才感覺(jué)會(huì)用了。
            測(cè)試了一下,debug下一萬(wàn)次回應(yīng)耗時(shí)800+毫秒,release下是200+毫秒,機(jī)器配置雙核2.5G英特爾,4個(gè)線(xiàn)程并行工作,無(wú)錯(cuò)的感覺(jué)真好,再也不用擔(dān)心iocp出一些奇怪的問(wèn)題啦,因?yàn)槭蔷奕藗儗?xiě)的實(shí)現(xiàn),呵呵。

            進(jìn)入正題,簡(jiǎn)要說(shuō)一下asio的實(shí)現(xiàn)原理吧。在win32平臺(tái)上,asio是基于IOCP技術(shù)實(shí)現(xiàn)的,我以前也用過(guò)IOCP,卻沒(méi)想到居然能擴(kuò)展成這樣,真是神奇!在其他平臺(tái)下還會(huì)有別的方法去實(shí)現(xiàn),具體見(jiàn)io_service類(lèi)下面這部分的源碼:
              // The type of the platform-specific implementation.
            #if defined(BOOST_ASIO_HAS_IOCP)
              typedef detail::win_iocp_io_service impl_type;
              friend 
            class detail::win_iocp_overlapped_ptr;
            #elif defined(BOOST_ASIO_HAS_EPOLL)
              typedef detail::task_io_service
            <detail::epoll_reactor<false> > impl_type;
            #elif defined(BOOST_ASIO_HAS_KQUEUE)
              typedef detail::task_io_service
            <detail::kqueue_reactor<false> > impl_type;
            #elif defined(BOOST_ASIO_HAS_DEV_POLL)
              typedef detail::task_io_service
            <detail::dev_poll_reactor<false> > impl_type;
            #else
              typedef detail::task_io_service
            <detail::select_reactor<false> > impl_type;
            #endif
            這部分代碼其實(shí)就在boost::asio::io_service類(lèi)聲明中的最前面幾行,可以看見(jiàn)在不同平臺(tái)下,io_service類(lèi)的實(shí)現(xiàn)將會(huì)不同。很顯然,windows平臺(tái)下當(dāng)然是win_iocp_io_service類(lèi)為實(shí)現(xiàn)了(不過(guò)我一開(kāi)始還以為win_iocp_io_service是直接拿出來(lái)用的呢,還在疑惑這樣怎么有移植性呢?官方文檔也對(duì)該類(lèi)只字不提,其實(shí)我卡殼就是卡在這里了,差點(diǎn)就直接用這個(gè)類(lèi)了^_^!)。

            那么就分析一下win_iocp_io_service的代碼吧,這里完全是用IOCP來(lái)路由各種任務(wù),大家使用post來(lái)委托任務(wù),內(nèi)部調(diào)用的其實(shí)是IOCP的PostQueuedCompletionStatus函數(shù),然后線(xiàn)程們用run來(lái)接受任務(wù),內(nèi)部其實(shí)是阻塞在IOCP的GetQueuedCompletionStatus函數(shù)上,一旦有了任務(wù)就立即返回,執(zhí)行完后再一個(gè)循環(huán),繼續(xù)阻塞在這里等待下一個(gè)任務(wù)的到來(lái),這種設(shè)計(jì)思想堪稱(chēng)神奇,對(duì)線(xiàn)程、服務(wù)以及任務(wù)完全解耦,靈活度達(dá)到了如此高度,不愧為boost庫(kù)的東西!我只能有拜的份了...

            說(shuō)一下總體的設(shè)計(jì)思想,其實(shí)io_service就像是勞工中介所,而一個(gè)線(xiàn)程就是一個(gè)勞工,而調(diào)用post的模塊相當(dāng)于富人們,他們?nèi)ブ薪樗腥蝿?wù),而勞工們就聽(tīng)候中介所的調(diào)遣去執(zhí)行這些任務(wù),任務(wù)的內(nèi)容就寫(xiě)在富人們給你的handler上,也就是函數(shù)指針,指針指向具體實(shí)現(xiàn)就是任務(wù)的實(shí)質(zhì)內(nèi)容。其實(shí)在整個(gè)過(guò)程中,富人們都不知道是哪個(gè)勞工幫他們做的工作,只知道是中介所負(fù)責(zé)完成這些就可以了。這使得邏輯上的耦合降到了最低。不過(guò)這樣的比喻也有個(gè)不恰當(dāng)?shù)牡胤剑绻惨@樣比喻的話(huà),我只能說(shuō):其實(shí)勞工里面也有很多富人的^o^! 。很多勞工在完成任務(wù)的過(guò)程中自己也托給中介所一些任務(wù),然后這些任務(wù)很可能還是自己去完成。這也難怪,運(yùn)行代碼的總是這些線(xiàn)程,那么調(diào)用post的肯定也會(huì)有這些線(xiàn)程了,不過(guò)不管怎么說(shuō),如此循環(huán)往復(fù)可以解決問(wèn)題就行,比喻不見(jiàn)得就得恰當(dāng),任何事物之間都不可能完全相同,只要能闡述思想就行。

            最后還要說(shuō)明的一點(diǎn)就是:委托的任務(wù)其實(shí)可以設(shè)定執(zhí)行的時(shí)間的,很不錯(cuò)的設(shè)定,內(nèi)部實(shí)現(xiàn)則是通過(guò)定時(shí)器原理,GetQueuedCompletionStatus有一個(gè)等待時(shí)間的參數(shù)似乎被用在這方面,還有源碼中的定時(shí)器線(xiàn)程我并沒(méi)有過(guò)多的去理解,總之大體原理已基本掌握,剩下的就是使勁的用它了?。。?br>
            另外為了方便人交流,在這里插入一些代碼可能更容易讓人理解吧,
            下面這個(gè)是啟動(dòng)服務(wù)時(shí)的代碼:
            void ServerFramework::run()
            {
                boost::thread_group workers;
                
            for (uint32 i = 0; i < mWorkerCount; ++i)
                    workers.create_thread(
                        boost::bind(
            &boost::asio::io_service::run, &mIoService));
                workers.join_all();
            }

            在打開(kāi)前就得分配好任務(wù),否則線(xiàn)程們運(yùn)行起來(lái)就退出了,阻塞不住,任務(wù)的分配就交給open函數(shù)了,它是分配了監(jiān)聽(tīng)端口的任務(wù),一旦有了連接就會(huì)拋出一個(gè)任務(wù),其中一個(gè)線(xiàn)程就會(huì)開(kāi)始行動(dòng)啦。
            void ServerFramework::open(const String& address, const String& port, uint32 nWorkers /*= DEFAULT_WORKER_COUNT*/)
            {
                
            // Open the acceptor with the option to reuse the address (i.e. SO_REUSEADDR).
                boost::asio::ip::tcp::resolver resolver(mIoService);
                boost::asio::ip::tcp::resolver::query query(address, port);
                boost::asio::ip::tcp::endpoint endpoint 
            = *resolver.resolve(query);

                mAcceptor.open(endpoint.protocol());
                mAcceptor.set_option(boost::asio::ip::tcp::acceptor::reuse_address(
            true));
                mAcceptor.bind(endpoint);
                mAcceptor.listen();

                mNextConnection 
            = new Connection(this);
                mAcceptor.async_accept(mNextConnection
            ->getSocket(),
                    boost::bind(
            &ServerFramework::__onConnect, this,
                    boost::asio::placeholders::error));

                mWorkerCount 
            = nWorkers;
                
            if (mWorkerCount == DEFAULT_WORKER_COUNT)
                
            {
                    mWorkerCount 
            = 4;
                }

            }

            open函數(shù)中給io_service的一個(gè)任務(wù)就是在有鏈接訪(fǎng)問(wèn)服務(wù)器端口的情況下執(zhí)行ServerFramework::__onConnect函數(shù),有一點(diǎn)需要格外注意的,io_service必須時(shí)刻都有任務(wù)存在,否則線(xiàn)程io_service::run函數(shù)將返回,于是線(xiàn)程都會(huì)結(jié)束并銷(xiāo)毀,程序?qū)⑼顺?,所以,你必須保證無(wú)論何時(shí)都有任務(wù)存在,這樣線(xiàn)程們即使空閑了也還是會(huì)繼續(xù)等待,不會(huì)銷(xiāo)毀。所以,我在ServerFramework::__onConnect函數(shù)中又一次給了io_service相同的任務(wù),即:繼續(xù)監(jiān)聽(tīng)端口,有鏈接了還是調(diào)用ServerFramework::__onConnect函數(shù)。如果你在ServerFramework::__onConnect執(zhí)行完了還沒(méi)有給io_service任務(wù)的話(huà),那么一切都晚了...... 代碼如下:
            void ServerFramework::__onConnect(const BoostSysErr& e)
            {
                
            if (e)
                
            {
                    MOELOG_DETAIL_WARN(e.message().c_str());
                }


                Connection
            * p = mNextConnection;
                mNextConnection 
            = new Connection(this);

                
            // 再次進(jìn)入監(jiān)聽(tīng)狀態(tài)
                mAcceptor.async_accept(mNextConnection->getSocket(),
                    boost::bind(
            &ServerFramework::__onConnect, this,
                    boost::asio::placeholders::error));

                
            // 處理當(dāng)前鏈接
                __addConnection(p);
                p
            ->start();
            }

            最后,展示一下這個(gè)類(lèi)的所有成員變量吧:
                // 用于線(xiàn)程池異步處理的核心對(duì)象
                boost::asio::io_service mIoService;

                
            // 網(wǎng)絡(luò)鏈接的接收器,用于接收請(qǐng)求進(jìn)入的鏈接
                boost::asio::ip::tcp::acceptor mAcceptor;

                
            // 指向下一個(gè)將要被使用的鏈接對(duì)象
                Connection* mNextConnection;

                
            // 存儲(chǔ)服務(wù)器鏈接對(duì)象的容器
                ConnectionSet mConnections;

                
            //// 為鏈接對(duì)象容器準(zhǔn)備的strand,防止并行調(diào)用mConnections
                //boost::asio::io_service::strand mStrand_mConnections;

                
            // 為鏈接對(duì)象容器準(zhǔn)備的同步鎖,防止并行調(diào)用mConnections
                boost::mutex mMutex4ConnSet;

                
            // 為控制臺(tái)輸出流準(zhǔn)備的strand,防止并行調(diào)用std::cout
                AsioService::strand mStrand_ConsoleIostream;

                
            // 工作線(xiàn)程的數(shù)量
                uint32 mWorkerCount;



            但愿這篇隨筆也能對(duì)正在研究asio的朋友們有所幫助吧。

            posted on 2009-06-26 22:14 釀妹汁 閱讀(9927) 評(píng)論(15)  編輯 收藏 引用 所屬分類(lèi): C++

            評(píng)論

            # re: ASIO攻破?。?! 2009-06-27 09:40 zhaoyang0618

            感謝!也在學(xué)習(xí)ASIO。
            能否簡(jiǎn)單的說(shuō)一下如何在A(yíng)SIO中使用thread_group呢?或者說(shuō)在A(yíng)SIO中使用線(xiàn)城池。  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?! 2009-06-30 17:15 亨德列克

            高  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。。? 2009-07-04 16:03

            @zhaoyang0618
            其實(shí)asio將線(xiàn)程與框架分離了開(kāi)來(lái),但是會(huì)在內(nèi)部輔助你進(jìn)行同步調(diào)用,完全不用你去擔(dān)心。你只要在線(xiàn)程的主函數(shù)中調(diào)用io_service::run方法就可以了,或者干脆將io_service::run當(dāng)成線(xiàn)程的主函數(shù)也行,就像我所用的方法一樣:
            boost::thread_group workers;
            for (uint32 i = 0; i < mWorkerCount; ++i)
            workers.create_thread(
            boost::bind(&boost::asio::io_service::run, &mIoService));
            workers.join_all();  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?! 2009-07-04 16:26

            @zhaoyang0618
            博客正文已經(jīng)添加了新的內(nèi)容,還望能指正。  回復(fù)  更多評(píng)論   

            # re: ASIO攻破!!! 2009-07-04 17:06 99網(wǎng)上書(shū)店

            開(kāi)源軟件的領(lǐng)袖和先驅(qū),從來(lái)沒(méi)有領(lǐng)過(guò)一天工資  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?! 2009-07-05 23:27 99書(shū)城

            能否簡(jiǎn)單的說(shuō)一下如何在A(yíng)SIO中使用thread_group呢  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?![未登錄](méi) 2009-09-02 21:17 ace

            喜歡用ace  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。。? 2009-10-27 16:14 dc

            為什么我一直覺(jué)得asio似乎并沒(méi)有將io/work分開(kāi)呢?
            按照這種async模型的話(huà),只要work卡住不去執(zhí)行async_recv的話(huà)一直無(wú)法從網(wǎng)絡(luò)緩沖區(qū)中提取出packet,最終可能導(dǎo)致緩沖區(qū)飽和?  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。。? 2009-10-27 16:16 dc

            不是數(shù)據(jù)包緩沖區(qū)..是io消息列隊(duì)..  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?! 2010-11-05 16:41 冬瓜

            為了防止io_service.run退出,可以增加一個(gè)work,這樣它就可以使它一直處理boost::asio::io_service::work  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?![未登錄](méi) 2012-08-30 13:21 jason

            你好,能給我一份關(guān)于boost的IOCP,完整的代碼嗎?對(duì)我而言,特別重要,希望大蝦幫助。  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?![未登錄](méi) 2012-08-30 13:23 jason

            我的郵件是zxl200406@163.com
            謝謝您的幫助。  回復(fù)  更多評(píng)論   

            # re: ASIO攻破!??! 2012-09-18 23:05 shanoa

            @jason
            boost源碼里有的呀,你上www.boost.org下載boost源碼吧  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。?![未登錄](méi) 2013-07-10 16:34 stonexin

            mAcceptor.bind(endpoint);

            這一行代碼最好做個(gè)判斷,加上error_code.不然ip地址或者端口不正確,會(huì)導(dǎo)致意外的崩潰。
            boost::system::error_code ec;
            mAcceptor.bind(endpoint,ec);
            if(ec)
            {
            std::cout<<"error: "<<ec.message()<<std::endl;
            }  回復(fù)  更多評(píng)論   

            # re: ASIO攻破?。。?a name="Post"> 2013-08-16 17:24 feelworld

            你好 能給一份實(shí)現(xiàn)這個(gè)框架的代碼嗎 不勝感激 276090345@qq.com  回復(fù)  更多評(píng)論   

            久久男人Av资源网站无码软件 | 久久久久久亚洲Av无码精品专口 | 好属妞这里只有精品久久| 国内精品久久国产| 亚洲精品乱码久久久久66| 久久综合给久久狠狠97色 | 香蕉久久久久久狠狠色| 亚洲综合日韩久久成人AV| …久久精品99久久香蕉国产| 久久久久亚洲精品中文字幕| 人妻精品久久无码区| 久久青青草原精品国产不卡 | 香港aa三级久久三级| 亚洲欧美日韩久久精品| www性久久久com| 怡红院日本一道日本久久| 国产99久久久国产精品小说| 国产激情久久久久影院小草 | 国产免费久久久久久无码| 亚洲精品乱码久久久久久蜜桃不卡| 情人伊人久久综合亚洲| 午夜人妻久久久久久久久| 久久久综合香蕉尹人综合网| 精品久久久久中文字幕日本| 热久久最新网站获取| 欧美大战日韩91综合一区婷婷久久青草| 无码超乳爆乳中文字幕久久| 亚洲婷婷国产精品电影人久久| 久久99精品国产99久久| 青青草原精品99久久精品66 | 久久综合综合久久97色| 日韩久久久久久中文人妻 | 2019久久久高清456| 久久综合久久伊人| 色偷偷91久久综合噜噜噜噜| 久久精品国产72国产精福利| 99久久精品免费看国产一区二区三区 | 一本色综合网久久| 久久无码专区国产精品发布| 亚洲综合久久夜AV | 一本大道久久东京热无码AV|