• <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 閱讀(230) 評論(0)  編輯 收藏 引用

            公告

            導航

            統計

            常用鏈接

            隨筆分類(178)

            3D游戲編程相關鏈接

            搜索

            最新評論

            国产69精品久久久久APP下载| 久久青青国产| 久久国产欧美日韩精品| 精品久久久久久国产91| 少妇久久久久久被弄到高潮| 欧美一区二区三区久久综合| 久久有码中文字幕| 性欧美丰满熟妇XXXX性久久久 | 9999国产精品欧美久久久久久| 色欲综合久久躁天天躁| 亚洲级αV无码毛片久久精品 | 久久成人小视频| www.久久热| 国内精品久久久久伊人av| 久久久久久国产精品美女| 久久久久亚洲AV成人网人人网站 | 久久AV高潮AV无码AV| 97超级碰碰碰久久久久| 中文字幕久久亚洲一区| 久久人人爽人人爽人人片AV麻豆 | 久久久久亚洲AV成人网| 国产精自产拍久久久久久蜜| 国产精品伦理久久久久久| 77777亚洲午夜久久多人| 99久久精品国产一区二区 | 久久精品人人做人人爽97| 久久无码精品一区二区三区| 久久99精品国产99久久6男男| 国内精品久久久久影院老司| 久久久久亚洲AV综合波多野结衣| 久久精品国产69国产精品亚洲| 久久久国产99久久国产一| 亚洲va久久久噜噜噜久久天堂| 日韩久久无码免费毛片软件| 久久久久国色AV免费观看| 精品欧美一区二区三区久久久 | 麻豆久久久9性大片| 热RE99久久精品国产66热| 久久精品国产亚洲AV不卡| 久久久99精品成人片中文字幕| 国产精品美女久久久久av爽|