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

            那誰的技術博客

            感興趣領域:高性能服務器編程,存儲,算法,Linux內核
            隨筆 - 210, 文章 - 0, 評論 - 1183, 引用 - 0
            數據加載中……

            讓libevent支持多線程

            libevent的使用方式是最開始調用event_init初始化一個全局的event_base指針,以后使用其中的API添加新的事件均是對這個指針進行的操作.

            試想如下一種典型的場景:主線程使用libevent處理網絡IO事件,接收新連接以及接收完客戶端的數據之后將該事件交給輔助線程進行處理,輔助線程處理完了,需要往客戶端發送回應數據,則再通過libevent提供的API將這個事件添加到可讀事件中.但是,由于libevent中這個event_base指針是全局的,如果多線程同時添加可能會造成多線程不安全問題,這個在libevent的代碼注釋里面作者也做了注解.

            在這個地方有人給出了解決的方案:
            已經有現成的解決方案:
            http://monkeymail.org/archives/libevent-users/2006-October/000257.html

            原作者給的地址好像已經失效了。iunknownspserver 中就是使用了他的代碼,可以從這里獲得
            http://code.google.com/p/spserver/source/browse/trunk/spserver/event_msgqueue.h
            (對應的C文件可以通過修改上面文件的后綴名所得).

            簡單的說一說這個event_msgqueue的原理及使用方式:
            它的結構體為:
            struct event_msgqueue {
               
            int push_fd;
               
            int pop_fd;
               
            int unlock_between_callbacks;

               
            struct event queue_ev;

               sp_thread_mutex_t 
            lock;
               
            void (*callback)(void *void *);
               
            void *cbarg;
               
            struct circqueue *queue;
            };
            其中的queue是保存事件的隊列,callback是處理事件的回調函數, lock是線程鎖.

            首先,它創建了一對socketpair(也就是結構體中的push_fd和pop_fd),將其中的一個通過libevent的event_add接口添加到關注的事件中,它的事件類型的是READ|PERSIST, 也就是說可讀同時永不刪除,而這個事件的回調函數是msgqueue_pop,這個函數的功能是將當前queue中的數據一一取出并且調用callback函數進行回調處理.
            其次,外部的多線程需要往libevent添加事件時使用這個文件提供的msgqueue_push函數進行添加,此時, 往socketpair中發送一個字節的數據, 這么做的目的是為了讓第一步已經添加的socketpair得到響應,此時, 由第一步,必然會觸發之前注冊的回調函數msgqueue_pop,將這個隊列中的事件取出來進行處理.

            這里使用socketpair起到了一個通知的作用,當隊列中有數據時,通過向socketpair發送數據以調用msgqueue_pop,將數據取出來進行處理.
            可見,這個event_msgqueue起了一個中間層的作用,輔助線程要往libevent添加事件就通過這個隊列,當隊列中有數據時再觸發libevent的事件處理機制進行處理.設計的非常巧妙.

            感謝iunknown兄的指點:)


            posted on 2009-01-12 14:49 那誰 閱讀(23609) 評論(6)  編輯 收藏 引用 所屬分類: 網絡編程服務器設計libevent

            評論

            # re: 讓libevent支持多線程  回復  更多評論   

            呵呵,你看一下libevent中對信號的處理,和這種方式是一模一樣的。Again,這也是很多事件處理框架所采用的方式(pipe或socketpair),即把多種事件源最后都統一到一種事件上去了——在Linux下,通常就是epoll處理的文件事件(socket也是文件)。
            2009-01-19 14:46 | Joshua Zhu

            # re: 讓libevent支持多線程  回復  更多評論   

            你 能給出 event_msgqueue的一個完整例子嗎?我的email:dragzhb@yahoo.com.cn,能發到我的郵箱嗎?多謝!
            2010-01-21 15:17 | hbzhang

            # re: 讓libevent支持多線程  回復  更多評論   

            1樓說的對啊,這個就是libevent對signal的處理方式。
            libevent就是講io事件、定時器和信號等事件統一到另外一件事去了。
            這里用的到的也是這個思想。
            2011-04-22 15:52 | zhanglistar

            # re: 讓libevent支持多線程  回復  更多評論   

            我需要完整的代碼啊 初學者 光看這些 還真不懂~~
            2013-03-13 17:54 | Metre

            # re: 讓libevent支持多線程  回復  更多評論   

            您好http://code.google.com/p/spserver/source/browse/trunk/spserver/event_msgqueue.h 好像失效了能否發我郵箱729720390@qq.com 謝謝!
            2015-03-06 11:53 | ruanquanfu

            # re: 讓libevent支持多線程  回復  更多評論   

            剛開始以為有個新的方法可以實現多線程。。。。其實就試類似pipe的方式, memcache就是這樣做的,可以參考一下
            2015-05-11 23:12 | fly2010love
            97精品依人久久久大香线蕉97| 成人精品一区二区久久| 精品国产乱码久久久久软件| 久久久SS麻豆欧美国产日韩| 狠狠色婷婷久久一区二区 | 久久亚洲中文字幕精品一区| 久久久久久亚洲精品不卡 | 久久人人爽人人爽人人片AV东京热 | 久久久久国产一级毛片高清版| 精品人妻伦九区久久AAA片69| 亚洲国产成人久久一区久久| 精品无码久久久久久午夜| 国产午夜精品久久久久九九电影| 久久久亚洲欧洲日产国码是AV| 亚洲欧美精品伊人久久| 亚洲午夜无码久久久久| 久久久国产精品| 97精品伊人久久久大香线蕉| 狠狠色婷婷久久综合频道日韩 | 久久亚洲AV成人无码国产| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 亚洲午夜久久久精品影院| 人妻丰满AV无码久久不卡 | 久久精品夜夜夜夜夜久久| 亚州日韩精品专区久久久| 91精品国产高清久久久久久国产嫩草| 亚洲国产精品无码久久| 久久九九久精品国产免费直播| 久久免费香蕉视频| 久久精品免费大片国产大片| 久久ZYZ资源站无码中文动漫| 午夜精品久久久久成人| 九九热久久免费视频| 香蕉久久一区二区不卡无毒影院| 精品国产VA久久久久久久冰| 中文国产成人精品久久不卡| 国产69精品久久久久APP下载| 亚洲а∨天堂久久精品| 青青草原综合久久大伊人导航 | 久久久亚洲AV波多野结衣 | 久久99精品久久久久久9蜜桃|