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

            lxyfirst

            C++博客 首頁(yè) 新隨筆 聯(lián)系 聚合 管理
              33 Posts :: 3 Stories :: 27 Comments :: 0 Trackbacks

            #

            我們知道ET模式下的事件觸發(fā)需要遇到EAGAIN結(jié)束,那么在tcp服務(wù)中不管是listen socket還是data socket,都需要做相應(yīng)處理,特別是listen socket ,很容易被遺漏 , 新連接未處理,從而出現(xiàn)close_wait狀態(tài)的socket 。
            另外錯(cuò)誤識(shí)別上,EAGAIN和EINTR一定要處理好,如果是主動(dòng)連接的data socket 在non-block下還要處理EINPROGRESS 。
            listen socket 處理
                while(1)
                {
                    
            int cfd = accept(GetFD(),(sa_t*)&caddr,&clen) ;
                    
            if ( cfd  == -1  ) 
                    {
                        
            if ( errno == EAGAIN ) return 0 ;
                        //other else  continue or close the socket
                        //
            else if ( errno == EINTR ) continue ;
                        //return -1 ;
                    }
                    set_nonblock(cfd) ;
                    
            //set_sock_nodelay(cfd) ;
                    
            //set_sock_linger(cfd) ;
                    if ( OnNewClient(cfd) != 0 )
                    {
                        close(cfd) ;
                    }
                }


            posted @ 2008-12-28 11:38 star 閱讀(711) | 評(píng)論 (0)編輯 收藏

            根據(jù)epoll的manual , epoll在ET模式下觸發(fā)可讀信號(hào)后必須一直讀取直到出現(xiàn)EAGAIN ,一般在應(yīng)用中會(huì)預(yù)先分布至少足夠一個(gè)完整報(bào)文大小的緩沖區(qū),接收到可讀信號(hào)后可以一次性讀取所有數(shù)據(jù),但是當(dāng)單個(gè)報(bào)文足夠大時(shí),很可能出現(xiàn)系統(tǒng)緩沖空間不足,從之導(dǎo)致無(wú)法一次性讀取所有數(shù)據(jù),雖然數(shù)據(jù)已經(jīng)到達(dá) 。也就是說(shuō),一個(gè)數(shù)據(jù)可讀信號(hào)觸發(fā)的讀取操作很可能無(wú)法讀取所有數(shù)據(jù),需要循環(huán)讀取直到遇到EAGAIN 。

            while(1)
            {
                
            int ret = read(fd,buf+pos,sizeof(buf)-pos) ;
                
            if ( ret == 0 )
                {   
            // client disconnect
                }
                
            else if ( ret < 0 )
                {
                    
            if ( errno == EAGAIN ) break ;
                    
            else if ( errno == EINTR  ) continue ;
                    
            // some error occurs
                }
                pos 
            += ret ;
                
            //check if buffer full or do request
                //**************note : sometimes the FIN and data arrive at the same time
            }
            // do request

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

            [
            215745][tcp_handler.cpp:59]read  fd=8 num=3426
            [
            215745][tcp_handler.cpp:61]total read  fd=8 num=3426      //一次完成讀取所有數(shù)據(jù)
            [
            215750][tcp_handler.cpp:53]disconnect  fd=8 
            [
            215751][server_app.cpp:169]new connection , fd=8
            [
            215751][tcp_handler.cpp:59]read  fd=8 num=57538
            [
            215751][tcp_handler.cpp:59]read  fd=8 num=3426
            [
            215751][tcp_handler.cpp:61]total read  fd=8 num=60964    //兩次才完成讀取所有數(shù)據(jù)
            [
            215756][tcp_handler.cpp:53]disconnect  fd=8 
            [
            215758][server_app.cpp:169]new connection , fd=8
            [
            215758][tcp_handler.cpp:59]read  fd=8 num=57538
            [
            215758][tcp_handler.cpp:59]read  fd=8 num=3426
            [
            215758][tcp_handler.cpp:61]total read  fd=8 num=60964
            [
            215803][tcp_handler.cpp:53]disconnect  fd=8 
            [
            215806][server_app.cpp:169]new connection , fd=8
            [
            215806][tcp_handler.cpp:59]read  fd=8 num=57538
            [
            215806][tcp_handler.cpp:59]read  fd=8 num=3426
            [
            215806][tcp_handler.cpp:61]total read  fd=8 num=60964
            [
            215811][tcp_handler.cpp:53]disconnect  fd=8

            posted @ 2008-12-28 11:24 star 閱讀(2932) | 評(píng)論 (2)編輯 收藏

            lua作為小巧精悍的腳本語(yǔ)言,易于嵌入c/c++中 , 廣泛應(yīng)用于游戲AI ,實(shí)際上在任何經(jīng)常變化的邏輯上都可以使用lua實(shí)現(xiàn),配合c/c++實(shí)現(xiàn)的底層接口服務(wù),能夠大大降低系統(tǒng)的維護(hù)成本。下面對(duì)lua和c/c++的交互調(diào)用做一個(gè)實(shí)例分析:
            lua提供了API用于在c/c++中構(gòu)造lua的運(yùn)行環(huán)境,相關(guān)接口如下:
            //創(chuàng)建lua運(yùn)行上下文
            lua_State* luaL_newstate(void) ;
            //加載lua腳本文件
            int luaL_loadfile(lua_State *L, const char *filename);

            lua和c/c++的數(shù)據(jù)交互通過(guò)"棧"進(jìn)行 ,操作數(shù)據(jù)時(shí),首先將數(shù)據(jù)拷貝到"棧"上,然后獲取數(shù)據(jù),棧中的每個(gè)數(shù)據(jù)通過(guò)索引值進(jìn)行定位,索引值為正時(shí)表示相對(duì)于棧底的偏移索引,索引值為負(fù)時(shí)表示相對(duì)于棧頂?shù)钠扑饕?,索引值?或-1為起始值,因此棧頂索引值永遠(yuǎn)為-1 ,棧底索引值永遠(yuǎn)為1 。 "棧"相當(dāng)于數(shù)據(jù)在lua和c/c++之間的中轉(zhuǎn)地。每種數(shù)據(jù)都有相應(yīng)的存取接口 。
            數(shù)據(jù)入"棧"接口:
            void  (lua_pushnil) (lua_State *L);
            void  (lua_pushnumber) (lua_State *L, lua_Number n);
            void  (lua_pushinteger) (lua_State *L, lua_Integer n);
            void  (lua_pushlstring) (lua_State *L, const char *s, size_t l);
            void  (lua_pushstring) (lua_State *L, const char *s);
            void  (lua_pushboolean) (lua_State *L, int b);
            void  (lua_pushcclosure) (lua_State *L, lua_CFunction fn, int n);

            數(shù)據(jù)獲取接口:
            lua_Number      (lua_tonumber) (lua_State *L, int idx);
            lua_Integer     (lua_tointeger) (lua_State *L, int idx);
            int             (lua_toboolean) (lua_State *L, int idx);
            const char     *(lua_tolstring) (lua_State *L, int idx, size_t *len);
            lua_CFunction   (lua_tocfunction) (lua_State *L, int idx);


            "棧"操作接口:
            int   (lua_gettop) (lua_State *L);
            void  (lua_settop) (lua_State *L, int idx);
            void  (lua_pushvalue) (lua_State *L, int idx);
            void  (lua_remove) (lua_State *L, int idx);
            void  (lua_insert) (lua_State *L, int idx);
            void  (lua_replace) (lua_State *L, int idx);
            int   (lua_checkstack) (lua_State *L, int sz);

            lua中定義的變量和函數(shù)存放在一個(gè)全局table中,索引值為L(zhǎng)UA_GLOBALSINDEX ,table相關(guān)操作接口:
            void  (lua_gettable) (lua_State *L, int idx);
            void  (lua_getfield) (lua_State *L, int idx, const char *k);
            void  (lua_settable) (lua_State *L, int idx);
            void  (lua_setfield) (lua_State *L, int idx, const char *k);

            當(dāng)"棧"中包含執(zhí)行腳本需要的所有要素(函數(shù)名和參數(shù))后,調(diào)用lua_pcall執(zhí)行腳本:
            int   (lua_pcall) (lua_State *L, int nargs, int nresults, int errfunc);

            下面進(jìn)行實(shí)例說(shuō)明:
            func.lua
            --變量定義
            width
            =1 ;
            height
            =2 ;
            --lua函數(shù)定義,實(shí)現(xiàn)加法
            function sum(a,b)
                
            return a+b ;
            end
            --lua函數(shù)定義,實(shí)現(xiàn)字符串相加
            function mystrcat(a,b)
                
            return a..b ;
            end
            --lua函數(shù)定義,通過(guò)調(diào)用c代碼中的csum函數(shù)實(shí)現(xiàn)加法
            function mysum(a,b)
                
            return csum(a,b) ;
            end

            test_lua.c
            #include <stdio.h>
            #include 
            <stdlib.h>
            #include 
            <string.h>
            #include <errno.h>
            //lua頭文件
            #include <lua.h>
            #include 
            <lualib.h>
            #include 
            <lauxlib.h>


            #define err_exit(num,fmt,args)  \
                
            do{printf("[%s:%d]"fmt"\n",__FILE__,__LINE__,##args);exit(num);} while(0)
            #define err_return(num,fmt,args)  \
                
            do{printf("[%s:%d]"fmt"\n",__FILE__,__LINE__,##args);return(num);} while(0)

            //lua中調(diào)用的c函數(shù)定義,實(shí)現(xiàn)加法
            int csum(lua_State* l)
            {
                
            int a = lua_tointeger(l,1) ;
                
            int b = lua_tointeger(l,2) ;
                lua_pushinteger(l,a
            +b) ;
                
            return 1 ;
            }

            int main(int argc,char** argv)
            {
                lua_State 
            * l = luaL_newstate() ;        //創(chuàng)建lua運(yùn)行環(huán)境
                
            if ( l == NULL ) err_return(-1,"luaL_newstat() failed"); 
                
            int ret = 0 ;
                ret 
            = luaL_loadfile(l,"func.lua") ;      //加載lua腳本文件
                
            if ( ret != 0 ) err_return(-1,"luaL_loadfile failed") ;
                ret 
            = lua_pcall(l,0,0,0) ;
                
            if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;

                lua_getglobal(l,
            "width");              //獲取lua中定義的變量
                lua_getglobal(l,
            "height");
                printf(
            "height:%ld width:%ld\n",lua_tointeger(l,-1),lua_tointeger(l,-2)) ;
                lua_pop(l,
            1) ;                        //恢復(fù)lua的棧

                
            int a = 11 ;
                
            int b = 12 ;
                lua_getglobal(l,
            "sum");               //調(diào)用lua中的函數(shù)sum
                lua_pushinteger(l,a) ;
                lua_pushinteger(l,b) ;
                ret 
            = lua_pcall(l,2,1,0) ;
                
            if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
                printf(
            "sum:%d + %d = %ld\n",a,b,lua_tointeger(l,-1)) ;
                lua_pop(l,
            1) ;

                
            const char str1[] = "hello" ;
                
            const char str2[] = "world" ;
                lua_getglobal(l,
            "mystrcat");          //調(diào)用lua中的函數(shù)mystrcat
                lua_pushstring(l,str1) ;
                lua_pushstring(l,str2) ;
                ret 
            = lua_pcall(l,2,1,0) ;
                
            if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
                printf(
            "mystrcat:%s%s = %s\n",str1,str2,lua_tostring(l,-1)) ;
                lua_pop(l,
            1) ;

                lua_pushcfunction(l,csum) ;         //注冊(cè)在lua中使用的c函數(shù)
                lua_setglobal(l,
            "csum") ;           //綁定到lua中的名字csum

                lua_getglobal(l,
            "mysum");           //調(diào)用lua中的mysum函數(shù),該函數(shù)調(diào)用本程序中定義的csum函數(shù)實(shí)現(xiàn)加法
                lua_pushinteger(l,a) ;
                lua_pushinteger(l,b) ;
                ret 
            = lua_pcall(l,2,1,0) ;
                
            if ( ret != 0 ) err_return(-1,"lua_pcall failed:%s",lua_tostring(l,-1)) ;
                printf(
            "mysum:%d + %d = %ld\n",a,b,lua_tointeger(l,-1)) ;
                lua_pop(l,
            1) ;

                lua_close(l) ;                     //釋放lua運(yùn)行環(huán)境
                
            return 0 ;
            }



            posted @ 2008-10-29 14:37 star 閱讀(28596) | 評(píng)論 (12)編輯 收藏

            僅列出標(biāo)題
            共4頁(yè): 1 2 3 4 
            综合久久精品色| 婷婷久久五月天| 亚洲国产精品久久久久网站| 久久国产精品一国产精品金尊| 久久综合88熟人妻| 国产精品99久久久久久猫咪| 亚洲欧美日韩精品久久亚洲区| 人妻无码中文久久久久专区| 久久国产V一级毛多内射| 模特私拍国产精品久久| 成人国内精品久久久久一区| 国产一区二区精品久久凹凸| 亚洲女久久久噜噜噜熟女| 国内精品久久久久久久97牛牛| 国产精品欧美久久久久无广告| 婷婷国产天堂久久综合五月| 国产精品久久久久aaaa| 欧美精品福利视频一区二区三区久久久精品| 伊人久久大香线蕉综合热线| 青青草国产成人久久91网| 久久久精品国产亚洲成人满18免费网站 | 国产精品久久国产精麻豆99网站| 久久国产高清一区二区三区| 久久精品青青草原伊人| 激情综合色综合久久综合| 亚洲国产精品无码久久久秋霞2| 中文字幕亚洲综合久久2| 色婷婷久久久SWAG精品| 久久精品国产影库免费看 | 久久99国产精品久久99果冻传媒| 亚洲国产精品综合久久网络| 久久精品免费观看| 天天爽天天狠久久久综合麻豆| 久久伊人色| 精品久久久久久无码人妻蜜桃| 国产成人久久精品区一区二区| 久久婷婷色综合一区二区| 免费一级做a爰片久久毛片潮| 国产综合免费精品久久久| 91精品国产91久久久久久蜜臀| 2022年国产精品久久久久|