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

            loop_in_codes

            低調(diào)做技術(shù)__歡迎移步我的獨(dú)立博客 codemaro.com 微博 kevinlynx

            自己實(shí)現(xiàn)memcached客戶端庫

            Kevin Lynx

            7.21.2008

            What's memcached ?

            memcached是一個(gè)以key-value的形式緩存數(shù)據(jù)的緩存系統(tǒng)。通過將數(shù)據(jù)緩存到內(nèi)存中,從而提高數(shù)據(jù)的獲取速度。
            memcached以key-value的形式來保存數(shù)據(jù),你可以為你每一段數(shù)據(jù)關(guān)聯(lián)一個(gè)key,然后以后可以通過這個(gè)key獲取
            這段數(shù)據(jù)。

            memcached是一個(gè)庫還是什么?memcached其實(shí)是一個(gè)單獨(dú)的網(wǎng)絡(luò)服務(wù)器程序。它的網(wǎng)絡(luò)底層基于libevent,你可以
            將其運(yùn)行在網(wǎng)絡(luò)中的一臺服務(wù)器上,通過網(wǎng)絡(luò),在遵循memcached的協(xié)議的基礎(chǔ)上與memcached服務(wù)器進(jìn)行通信。

            What do we want to wrap ?

            我們需要做什么?我們只需要遵循memcached的協(xié)議(參見該文檔),封裝網(wǎng)絡(luò)層的通信,讓上層可以通過調(diào)用諸如
            add/get之類的接口即可實(shí)現(xiàn)往memcached服務(wù)器緩存數(shù)據(jù),以及取數(shù)據(jù)。上層程序員根本不知道這些數(shù)據(jù)在網(wǎng)絡(luò)
            上存在過。

            這個(gè)東西,也就是memcached官方所謂的client apis。你可以使用現(xiàn)成的客戶端庫,但是你也可以將這種重造輪子
            的工作當(dāng)作一次網(wǎng)絡(luò)編程的練習(xí)。it's up to you.:D

            Where to start ?

            很遺憾,對于windows用戶而言,memcached官方?jīng)]有給出一個(gè)可以執(zhí)行或者可以直接F7即可得到可執(zhí)行文件的下載
            (如果你是vc用戶)。幸運(yùn)的是,已經(jīng)有人做了這個(gè)轉(zhuǎn)換工作。

            你可以從http://jehiah.cz/projects/memcached-win32/這里下載到memcached的windows版本,包括可執(zhí)行程序和
            源代碼。

            我們直接可以運(yùn)行memcached.exe來安裝/開啟memcached服務(wù)器,具體步驟在以上頁面有所提及:

            安裝:memcached.exe -d install,這會在windows服務(wù)里添加一個(gè)memcached服務(wù)
            運(yùn)行:memcached.exe 
            -d start,你也可以通過windows的服務(wù)管理運(yùn)行。

               
            然后,你可以在任務(wù)管理器里看到一個(gè)'memcached'的進(jìn)程,很占內(nèi)存,因?yàn)檫@是memcached。

            So, here we go ...

            通過以上步驟運(yùn)行的memcached,默認(rèn)在11211端口監(jiān)聽(是個(gè)TCP連接,可以通過netstat查看)。接下來,我們就可
            以connect到該端口上,然后send/recv數(shù)據(jù)了。發(fā)送/接收數(shù)據(jù)只要遵循memcached的協(xié)議格式,一切都很簡單。

            使用最簡單的阻塞socket連接memcached服務(wù)器:

                   SOCKET s = socket( AF_INET, SOCK_STREAM, 0 );
                    
            struct sockaddr_in addr;
                    memset( 
            &addr, 0sizeof( addr ) );
                    addr.sin_family 
            = AF_INET;
                    addr.sin_port 
            = htons( 11211 );
                    addr.sin_addr.s_addr 
            = inet_addr( "127.0.0.1" ); 

                    ret 
            = connect( s, (struct sockaddr*&addr, sizeof( addr ) );
                    
            if( ret == 0 )
                    
            {
                        printf( 
            "connect ok\n" );
                    }
             

                   
            About the protocol

            簡單地提一下memcached的協(xié)議。

            可以說,memcached的協(xié)議是基于行的協(xié)議,因?yàn)闊o論是客戶端請求還是服務(wù)器端應(yīng)答,都是以"\r\n"作為結(jié)束符。
            memcached的協(xié)議將數(shù)據(jù)(send/recv操作的數(shù)據(jù))分為兩種類型:命令和用戶數(shù)據(jù)。

            命令用于服務(wù)器和客戶端進(jìn)行交互;而用戶數(shù)據(jù),很顯然,就是用戶想要緩存的數(shù)據(jù)。

            關(guān)于用戶數(shù)據(jù),你只需要將其編碼成字節(jié)流(說白了,只要send函數(shù)允許即可),并附帶數(shù)據(jù)結(jié)束標(biāo)志"\r\n"發(fā)送即可。

            關(guān)于命令,memcached分為如下幾種命令:存儲數(shù)據(jù)、刪除數(shù)據(jù)、取出數(shù)據(jù)、其他一些獲取信息的命令。其實(shí)你換個(gè)角度
            想想,memcached主要就是將數(shù)據(jù)存儲到內(nèi)存里,所以命令也多不了哪去,基本就停留在add/get/del上。

            關(guān)于key,memcached用key來標(biāo)識數(shù)據(jù),每一個(gè)key都是一個(gè)不超過255個(gè)字符的字符串。

            到這里,你可以發(fā)現(xiàn)memcached對于數(shù)據(jù)的存儲方式(暴露給上層)多少有點(diǎn)像std::map,如果你愿意,你可以將客戶端
            API封裝成map形式。= =#

            具體實(shí)現(xiàn)

            接下來可以看看具體的實(shí)現(xiàn)了。

            首先看看存儲數(shù)據(jù)命令,存儲數(shù)據(jù)命令有:add/set/replace/append/prepend/cas。存儲命令的格式為:
            <command name> <key> <flags> <exptime> <bytes> [noreply]\r\n
            具體字段的含義參看protocol.txt文件,這里我對set舉例,如下代碼,阻塞發(fā)送即可:

                    char cmd[256] ;
                    
            char data[] = "test data";
                    sprintf( cmd, 
            "set TestKey 0 0 %d\r\n", strlen( data ) );
                    ret 
            = send( s, cmd, strlen( cmd ), 0 ); 


            注意:noreply選項(xiàng)對于有些memcached版本并不被支持,例如我們使用的1.2.2版本。注意官方的changelog即可。

            當(dāng)你發(fā)送了存儲命令后,memcached會等待客戶端發(fā)送數(shù)據(jù)塊。所以我們繼續(xù)發(fā)送數(shù)據(jù)塊:

             

                    ret = send( s, data, strlen( data ), 0 );
                    ret 
            = send( s, "\r\n"20 ); // 數(shù)據(jù)結(jié)束符

             

            然后,正常的話,memcached服務(wù)器會返回應(yīng)答信息給客戶端。

             

                    char reply[256];
                    ret 
            = recv( s, reply, sizeof( reply ) - 10 );
                    reply[ret] 
            = 0;
                    printf( 
            "server reply : %s\n", reply ); 

             

            如果存儲成功,服務(wù)器會返回STORED字符串。memcached所有應(yīng)答信息都是以字符串的形式給出的。所以可以直接printf出來。

            關(guān)于其他的操作,我就不在這里列舉例子了。我提供了我封裝的memcached客戶端庫的完整代碼下載,使用的是阻塞socket,
            對應(yīng)著memcached的協(xié)議看,很容易看懂的。

            It's a story about a programmer...

            最近發(fā)覺自己有點(diǎn)極端,要么寫純C的代碼,要么寫滿是template的泛型代碼。

             

            相關(guān)代碼下載

            posted on 2008-07-21 15:54 Kevin Lynx 閱讀(6369) 評論(5)  編輯 收藏 引用 所屬分類: network

            評論

            # re: 自己實(shí)現(xiàn)memcached客戶端庫 2008-10-23 16:56 浪跡天涯

            學(xué)習(xí)了!
            不知博主是否真實(shí)測試過采用memcached的服務(wù)器性能有多少提升?  回復(fù)  更多評論   

            # re: 自己實(shí)現(xiàn)memcached客戶端庫[未登錄] 2008-10-23 20:44 Kevin Lynx

            @浪跡天涯
            老實(shí)說,實(shí)際項(xiàng)目里還沒用過memcached。  回復(fù)  更多評論   

            # re: 自己實(shí)現(xiàn)memcached客戶端庫 2009-03-26 17:35 阿斯蒂芬

            你的代碼在linux上編譯報(bào)出這個(gè)錯(cuò)誤error: storage size of 'addr' isn't known
            行號在sizeof( addr ).也就是獲取結(jié)構(gòu)大小的時(shí)候.  回復(fù)  更多評論   

            # re: 自己實(shí)現(xiàn)memcached客戶端庫[未登錄] 2009-03-26 18:05 Kevin Lynx

            @阿斯蒂芬
            能不能具體點(diǎn)?哪個(gè)文件,哪一行?
            我稍微搜索了下代碼,發(fā)現(xiàn)只有在fun_test.txt里才有sizeof( addr ),這個(gè)文件是用于功能測試的,只要kl_memcached下的源文件編譯得過你就可以使用。:)  回復(fù)  更多評論   

            # re: 自己實(shí)現(xiàn)memcached客戶端庫 2011-06-21 11:23 110

            @阿斯蒂芬
            這是windows下的代碼,姐  回復(fù)  更多評論   

            久久精品免费观看| 久久久噜噜噜www成人网| 国产激情久久久久久熟女老人| 亚洲国产成人久久综合一 | 奇米综合四色77777久久| 精品综合久久久久久888蜜芽| 狠狠色婷婷久久一区二区三区| 欧美精品国产综合久久| 99久久精品国产一区二区| 亚洲欧美日韩精品久久亚洲区| 2019久久久高清456| 欧美日韩精品久久久久| 国内高清久久久久久| 久久精品国产清高在天天线| 久久午夜伦鲁片免费无码| 国产午夜免费高清久久影院| 亚洲国产成人久久精品动漫| 精品久久久久久久久久久久久久久| 少妇被又大又粗又爽毛片久久黑人| 日产精品久久久久久久性色| av无码久久久久久不卡网站| 国产激情久久久久影院小草 | 狠狠久久亚洲欧美专区| 2021国产成人精品久久| 久久人人爽人人澡人人高潮AV| 久久天天躁狠狠躁夜夜avapp| 波多野结衣中文字幕久久| 伊人久久免费视频| 欧美久久天天综合香蕉伊| 久久久www免费人成精品| 久久天天躁狠狠躁夜夜av浪潮| 久久亚洲AV成人无码| 97精品久久天干天天天按摩| 久久精品国产亚洲αv忘忧草 | 久久亚洲sm情趣捆绑调教| av国内精品久久久久影院 | 亚洲人成无码久久电影网站| 国产亚洲综合久久系列| 色老头网站久久网| 青青热久久国产久精品| 国产91久久精品一区二区|