• <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>
            隨筆 - 96  文章 - 255  trackbacks - 0
            <2010年6月>
            303112345
            6789101112
            13141516171819
            20212223242526
            27282930123
            45678910

            E-mail:zbln426@163.com QQ:85132383 長期尋找對戰(zhàn)略游戲感興趣的合作伙伴。

            常用鏈接

            留言簿(21)

            隨筆分類

            隨筆檔案

            SDL相關(guān)網(wǎng)站

            我的個人網(wǎng)頁

            我的小游戲

            資源下載

            搜索

            •  

            積分與排名

            • 積分 - 493195
            • 排名 - 39

            最新評論

            閱讀排行榜

            評論排行榜

            UDP的客戶端看起來幾乎就是服務(wù)器端的翻版,甚至比服務(wù)器端更簡單——因?yàn)椴恍枰猙ind()本機(jī)地址:
            class UDPClientSock: public BaseSock {
            protected:
                mutable sockaddr_in lastfromSockAddr;
                sockaddr_in destinationSockAddr;
                
            char* preBuffer;
                
            int preBufferSize;
                mutable 
            int preReceivedLength;
            public:
                
            explicit UDPClientSock(int pre_buffer_size = 32);
                
            virtual ~UDPClientSock();
                
            void UDPSetDest(const char* dest_IP,
                        
            const unsigned short& dest_port);
                
            void UDPSetDest(const sockaddr_in& dest_sock_addr);
                
            int UDPReceive() const;
                
            int UDPSendtoDest(const char* send_data,
                        
            const int& data_length) const;
            };
            在最初設(shè)計(jì)這個類的時候,我曾經(jīng)考慮過安排一個服務(wù)器地址的私有數(shù)據(jù)成員,并且在構(gòu)造函數(shù)里面指定服務(wù)器的地址。但是,后來我覺得使用“目的地”比“服務(wù)器”更加能體現(xiàn)出UDP無連接的本質(zhì)特點(diǎn)。TCP之所以有個服務(wù)器,是因?yàn)門CP的客戶端只能和自己的服務(wù)器端通訊。而UDP的客戶端可以與任何一個UDP端口通訊——只要知道對方的地址(IP地址和UDP端口)就可以發(fā)送數(shù)據(jù)包。況且,在網(wǎng)絡(luò)情況越來越復(fù)雜的今天,很多服務(wù)器都不僅僅使用一個IP地址或者域名,比如網(wǎng)站和游戲服務(wù)器,而對于客戶端來說,只是在意連接到了指定的網(wǎng)站,比如google,而并不清楚是連接到google的哪個服務(wù)器。程序內(nèi)部可能會根據(jù)網(wǎng)絡(luò)條件對具體連接的服務(wù)器地址進(jìn)行調(diào)整,所以,可以隨時根據(jù)具體情況指定“目的地”,而不是一開始就指定一個“服務(wù)器”地址,這種策略顯得更加靈活。
            通常情況下,客戶端也并不在意lastfromSockAddr,因?yàn)樽詈笠淮蝸硐虻牡刂罚褪悄康牡胤?wù)器的地址。我們說過,服務(wù)器的端口是指定的,這是為了讓客戶端明確的知道,可以去連接。而客戶端的端口的端口則是系統(tǒng)指定的——我們并沒有在客戶端調(diào)用bind(),所以socket機(jī)制會自動幫我們綁定一個端口。通常客戶端自己也不需要知道這個端口號是多少,只有接收到這次UDP數(shù)據(jù)報(bào)的服務(wù)器端知道,并且按照這個端口號將服務(wù)器的信息傳送過來——沒有收到這個端口發(fā)出的數(shù)據(jù)報(bào)的UDP端口很難知道這個系統(tǒng)指定的端口號是多少。但是,因?yàn)檫@個UDP端口實(shí)際上是可以接受來自其他任何UDP端口的數(shù)據(jù)的,所以,如果你需要驗(yàn)證發(fā)送某次數(shù)據(jù)的地址是不是你所期望的,比如是不是來自服務(wù)器,可能就會用到lastfromSockAddr。
            UDPClientSock::UDPClientSock(int pre_buffer_size):
            preBufferSize(pre_buffer_size), preReceivedLength(
            0)
            {
                preBuffer 
            = new char[preBufferSize];
                memset(
            &lastfromSockAddr, 0sizeof(lastfromSockAddr));
                memset(
            &destinationSockAddr, 0sizeof(destinationSockAddr));

                sockFD 
            = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);
                
            if (sockFD < 0) {
                    sockClass::error_info(
            "sock() failed.");
                }
            }

            UDPClientSock::
            ~UDPClientSock()
            {
                delete [] preBuffer;
                close(sockFD);
            }
            其它4個類方法,跟server端的簡直一模一樣。
            void UDPClientSock::UDPSetDest(const char* dest_IP,
                                           
            const unsigned short& dest_port)
            {
                destinationSockAddr.sin_family 
            = AF_INET;
                destinationSockAddr.sin_addr.s_addr 
            = inet_addr(dest_IP);
                destinationSockAddr.sin_port 
            = htons(dest_port);
            }

            void UDPClientSock::UDPSetDest(const sockaddr_in& dest_sock_addr)
            {
                destinationSockAddr.sin_family 
            = dest_sock_addr.sin_family;
                destinationSockAddr.sin_addr.s_addr 
            = dest_sock_addr.sin_addr.s_addr;
                destinationSockAddr.sin_port 
            = dest_sock_addr.sin_port;
            }

            int UDPClientSock::UDPReceive() const
            {
                socklen_t
             from_add_len = sizeof(lastfromSockAddr); //use int in win32
                preReceivedLength 
            = recvfrom(    sockFD,
                                                preBuffer,
                                                preBufferSize,
                                                
            0,
                                                (sockaddr
            *)&lastfromSockAddr,
                                                
            &from_add_len);
                
            if ( preReceivedLength < 0) {
                    sockClass::error_info(
            "recv() failed.");
                }

                
            return preReceivedLength;
            }

            int UDPClientSock::UDPSendtoDest(const char* send_data,
                                             
            const int& data_length) const
            {
                
            int send_message_size = sendto(    sockFD,
                                                send_data,
                                                data_length,
                                                
            0,
                                                (sockaddr
            *)&destinationSockAddr,
                                                
            sizeof(destinationSockAddr));
                
            if (send_message_size < 0) {
                    sockClass::error_info(
            "send() failed.");
                }
                
            if (send_message_size != data_length) {
                    sockClass::error_info(
                        
            "send() sent a different number of bytes than expected.");
                }
                
            return send_message_size;
            }
            posted on 2010-06-10 19:37 lf426 閱讀(1730) 評論(0)  編輯 收藏 引用 所屬分類: SDL入門教程socket 編程入門教程
            久久久久久国产a免费观看不卡| 久久99精品久久久久久| 青青青青久久精品国产h| 久久午夜电影网| 成人国内精品久久久久影院VR| 久久久久女教师免费一区| 伊人久久大香线蕉综合5g| 欧美精品久久久久久久自慰| 国产精品久久成人影院| 2021国产精品午夜久久| 久久精品国产亚洲AV高清热 | 欧洲性大片xxxxx久久久| 久久精品国产亚洲AV香蕉| 国内精品久久久久久久久电影网| 久久精品国产99国产精品导航| 国产成人久久精品区一区二区| 久久强奷乱码老熟女| 99久久亚洲综合精品成人| 久久国产高潮流白浆免费观看| 久久精品国产亚洲5555| 9久久9久久精品| 色8久久人人97超碰香蕉987| 亚洲国产天堂久久综合| 狠狠色丁香婷婷综合久久来来去 | 国内精品久久久久影院优| 一本色道久久综合亚洲精品| 伊人久久大香线蕉综合5g| 国产福利电影一区二区三区,免费久久久久久久精 | 久久伊人精品一区二区三区| 日韩欧美亚洲综合久久影院d3| 日产精品久久久久久久| 久久久久亚洲av成人网人人软件| 久久人妻少妇嫩草AV无码蜜桃| 久久综合狠狠综合久久激情 | 国产叼嘿久久精品久久| 国产福利电影一区二区三区,免费久久久久久久精 | 天天做夜夜做久久做狠狠| 国内精品伊人久久久久影院对白 | 日本加勒比久久精品| 久久人搡人人玩人妻精品首页| 久久99精品国产麻豆不卡|