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

            天行健 君子當自強而不息

            Getting Online with Multiplayer Gaming(10)

             

            DirectPlay Messages to Game Messages

            As I’ve mentioned before, the server needs to convert the DirectPlay network messages
            into the game-related messages you’ve just read about. You accomplish this by
            processing incoming player connection, disconnection, and receive data messages from
            DirectPlay and converting those messages into game messages.

            To accomplish this conversion of messages, you derive a class from cNetworkServer
            and override the create_player, destroy_player, and receive functions:

            class cServer : public cNetworkServer
            {
            protected:
                
            virtual bool create_player(const DPNMSG_CREATE_PLAYER* msg);
                
            virtual bool destroy_player(const DPNMSG_DESTROY_PLAYER* msg);
                
            virtual bool receive(const DPNMSG_RECEIVE* msg);
            };

            Because I’m using the System Core to handle application processing, a problem
            quickly arises when dealing with the network. The network component and application
            component are two separate entities, which means that neither component is
            allowed to modify the other's private data.

            As Figure 19.11 illustrates, the network component needs a way to siphon incoming
            messages into the application, which by chance is handled by creating three public
            functions that match the network class’s functions.

            To use the three message functions in the application component, you construct a
            derived cFramework class that contains the three public functions as follows:

            class cApp : public cFramework
            {
            private:
                HWND                m_controls[3];
                
                CRITICAL_SECTION    m_msg_cs;
                cMesh               m_level_mesh;

                GUID*               m_adapter_guid;
                cNetworkAdapter     m_adapter;
                cServer             m_server;

                
            long                m_connected_player_num; 
                sPlayer*            m_players;

                sMsg*               m_msgs;
                
            long                m_msg_head;
                
            long                m_msg_tail;

                
            ///////////////////////////////////////////////////////////////////////////////

            public:
                
            void set_adapter_guid(GUID* adapter_guid)
                {
                    m_adapter_guid = adapter_guid;
                }

            public:
                cApp();
                
                
            virtual bool init();
                
            virtual bool frame();
                
            virtual void shutdown();

                
            void create_player(const DPNMSG_CREATE_PLAYER* msg);
                
            void destroy_player(const DPNMSG_DESTROY_PLAYER* msg);
                
            void receive(const DPNMSG_RECEIVE* msg);

            private:
                
            bool select_adapter();
                
            void setup_app_window();
                
            bool init_game();
                
            bool host_game();    

                
            void list_players();

                
            void process_queue_msg();
                
            void update_players();
                
            void update_network();
                
            void update_latency();

                
            bool send_player_info(const sMsg* msg, DPNID to);
                
            bool queue_msg(const void* msg);
                
            bool add_player(const sMsg* msg);
                
            void remove_player(const sMsg* msg);
                
            bool player_state_change(const sMsg* msg);

                
            bool send_network_msg(void* msg, long send_flags, int to);

                
            bool check_intersect(cMesh* mesh,
                                     
            float x_start, float y_start, float z_start,
                                     
            float x_end,   float y_end,   float z_end);
            };

            To start sending DirectPlay messages to the application class, you code the overridden
            cServer functions to call upon the matching application functions. In order for the
            server to know which application class instance to send messages to, you need to
            declare a global variable that points to the current application class instance in use:

            cApp* g_app;
            cNetworkAdapter* g_adapter;

            Inside the derived application class’s constructor, you then point the global
            g_app variable to the application class instance:

            cApp::cApp()
            {    
                m_adapter_guid = NULL;
                m_msgs         = NULL;
                m_msg_head     = 0;
                m_msg_tail     = 0;

                m_connected_player_num = 0;
                m_players = NULL;

                g_app     = 
            this;
                g_adapter = &m_adapter;

                InitializeCriticalSection(&m_msg_cs);
            }

            Now, you can code the network server component to send incoming messages to
            the application object defined by the global g_app pointer:


            bool cServer::create_player(const DPNMSG_CREATE_PLAYER* msg)
            {
                g_app->create_player(msg);

                
            return true;
            }

            bool cServer::destroy_player(const DPNMSG_DESTROY_PLAYER* msg)
            {
                g_app->destroy_player(msg);

                
            return true;
            }

            bool cServer::receive(const DPNMSG_RECEIVE* msg)
            {
                g_app->receive(msg);

                
            return true;
            }

            The server component is now complete and is forwarding network messages to the
            application class. To convert those network messages to game-related messages,
            the application class must contain the following public functions:


            void cApp::create_player(const DPNMSG_CREATE_PLAYER* msg)
            {
                sCreatePlayerMsg create_msg;

                create_msg.header.type      = MSG_CREATE_PLAYER;
                create_msg.header.size      = 
            sizeof(sCreatePlayerMsg);
                create_msg.header.player_id = msg->dpnidPlayer;

                queue_msg(&create_msg);
            }

            ///////////////////////////////////////////////////////////////////////////////////////

            void cApp::destroy_player(const DPNMSG_DESTROY_PLAYER* msg)
            {
                sDestroyPlayerMsg destroy_msg;

                destroy_msg.header.type      = MSG_DESTROY_PLAYER;
                destroy_msg.header.size      = 
            sizeof(sDestroyPlayerMsg);
                destroy_msg.header.player_id = msg->dpnidPlayer;

                queue_msg(&destroy_msg);
            }

            ///////////////////////////////////////////////////////////////////////////////////////

            void cApp::receive(const DPNMSG_RECEIVE* msg)
            {
                sMsgHeader* header = (sMsgHeader*) msg->pReceiveData;

                
            // make sure it is a valid message type and queue it
                switch(header->type)
                {
                
            case MSG_SEND_PLAYER_INFO:
                
            case MSG_STATE_CHANGE:
                    queue_msg(msg->pReceiveData);
                    
            break;
                }
            }

            You can see that in each of the three functions, I’m constructing a game-related
            message using the data from the DirectPlay messages provided. When a player tries
            to connect to the server, a create-player message is created that stores the connecting
            player’s DirectPlayer identification number (along with the message type and size).
            That create-player message is then queued.

            As for players disconnecting from the game, a disconnect-player message is constructed
            and queued. Last, whenever data (other than a system message) is
            received from a client, the cApp::receive function checks it to see whether it’s a valid
            message type, and if so, the message is queued.

            I keep mentioning the message queue and how the previously shown function adds
            messages to the queue. Next, you find out what the queue is and how it works.

            posted on 2007-12-18 20:35 lovedday 閱讀(248) 評論(0)  編輯 收藏 引用

            公告

            導航

            統計

            常用鏈接

            隨筆分類(178)

            3D游戲編程相關鏈接

            搜索

            最新評論

            污污内射久久一区二区欧美日韩| 亚洲国产成人精品91久久久 | 婷婷久久综合九色综合98| 狠狠色丁香婷综合久久| 日韩十八禁一区二区久久| 国产69精品久久久久9999APGF | 久久精品极品盛宴观看| 久久精品国产99久久久| 久久最新免费视频| 精品免费tv久久久久久久| 亚洲国产精品成人久久蜜臀 | 7国产欧美日韩综合天堂中文久久久久| 亚洲欧美另类日本久久国产真实乱对白| 午夜精品久久久久久中宇| 久久久这里有精品中文字幕| 久久精品国产99国产电影网 | 国内精品久久久久| 久久婷婷国产综合精品| 一本久久综合亚洲鲁鲁五月天亚洲欧美一区二区 | 久久久久亚洲AV无码永不| 中文字幕无码av激情不卡久久| 国产精品久久久久久| 久久AV高潮AV无码AV| 亚洲欧美另类日本久久国产真实乱对白| 国产91色综合久久免费分享| 久久精品国产网红主播| 国内精品久久久久久久久电影网| 亚洲日韩欧美一区久久久久我| 久久久久久久亚洲精品| 精品国产一区二区三区久久蜜臀| 久久精品视频网| 99久久综合国产精品二区| 欧美久久精品一级c片片| 久久精品国产99国产电影网| 狠狠久久亚洲欧美专区| 国产午夜精品理论片久久| 久久精品国产亚洲av瑜伽| 日本高清无卡码一区二区久久| 久久精品人妻一区二区三区| 日韩AV毛片精品久久久| 国色天香久久久久久久小说|