[ZT] online-game server design (1, 2)

Source: Internet
Author: User
Before talking about this topic, let everyone know what a server is. In online games, servers play the role of synchronization, broadcast and server-initiated behaviors, such as weather, npc ai, etc, the reason why many online game servers need to carry some game logic operations is to prevent client cheating. With this understanding, this series of articles will be divided into two parts to talk about the design of online game servers. One part is about how to do a good job of network connection, synchronization, broadcast, and NPC settings for servers, the other part focuses on which logic is suitable for the server and what structure is used to arrange these logic.

Server network connection

Most online game servers use a non-blocking select structure. Why? Because the network game servers need to process a large number of connections, and most of them will choose to run in Linux/Unix, it is not cost-effective to open a thread for each user, on the one hand, threads in Linux/Unix are simulated by the concept of processes, which consume system resources. Besides I/O, basically, each thread has no redundant tasks that require parallel processing, and online games are highly interactive. Therefore, inter-thread synchronization becomes very troublesome. As a result, blocking is obviously unrealistic for a Single-threaded server that contains a large number of network connections. For network connections, a structure is required for storage, which must contain a buffer for writing messages to the client, and a buffer for reading messages from the client, the specific size depends on the message structure. In addition, for synchronization, it takes some time to proofread The values. Different values are also required to record the current status. The following shows a preliminary connection structure:

Typedef connection_s {

User_t * ob;/* point to the structure for processing the server logic */

Int FD;/* socket connection */

Struct sockaddr_in ADDR;/* connection address information */

Char text [max_text];/* Received message buffer */

Int text_end;/* tail pointer of the received message buffer */

Int text_start;/* Header pointer of the received message buffer */

Int last_time;/* when is the last message received */

Struct timeval latency;/* difference between the local time of the client and the local time of the server */

Struct timeval last_confirm_time;/* time of the last verification */

Short is_confirmed;/* Whether the connection has passed verification */

Int ping_num;/* The Ping value from the client to the server */

Int ping_ticker;/* how many I/O cycles are processed to update the ping value once */

Int message_length;/* length of the buffer message to be sent */

Char message_buf [max_text];/* Sending buffer */

Int iflags;/* The connection status */

} Connection_t;

The server processes all connections cyclically. It is an endless loop. Each loop uses select to check whether a new connection has arrived. Then, it loops through all connections to see which connection can be written or read, read and Write the connection. Since all the processing is non-blocking, All socket Io can be completed using one thread.

Due to the network transmission relationship, each Recv () may contain more than one message, or less than one message. How can this problem be solved? Therefore, two pointers are used for receiving message buffering. Each receipt starts from text_start, because the remaining half of the message received last time may exist, then text_end points to the end of the message buffer. In this way, two pointers can be used to easily handle this situation. In addition, it is worth noting that the process of parsing messages is a cyclical process, it is possible that more than two messages are received in the message buffer at a time. At this time, it should be executed until there is only one message that is not available in the message buffer. The general process is as follows:

While (text_end-text_start> A complete message length)

{

Start processing from text_start;

Text_start + = message length;

}

Memcpy (text, text + text_start, text_end-text_start );

For message processing, you first need to know the total messages in your game and all the messages to design a reasonable message header. Generally, messages can be divided into four parts: main message, scenario message, synchronous message, and interface message. The main message includes all the actions of the role controlled by the client, including walking, running, and fighting. Scenario messages include weather changes, and some things appear in the scenario for a certain period of time. The characteristics of such messages are that all message initiators are servers, broadcast objects are all players in the scenario. The synchronous message is intended for a player who initiates an object and is broadcast to all players who can see the object on the server. The message also includes all actions, different from the main message, this type of message is sent from the server to the client, and the main message is generally sent from the client to the server. Finally, the interface message includes the chat message sent from the server to the client and various attributes and status information.

The following describes the composition of a message. Generally, a message consists of a message header and a message body. The length of the message header remains unchanged, and the length of the message body is variable, the length of the message body to be saved in the message body. To clearly differentiate each message, you need to define a unique identifier of the message header and the type and ID of the message. The structure of the message header is as follows:

Type struct message_s {

Unsigned short message_sign;

Unsigned char message_type;

Unsigned short message_id

Unsigned char message_len

} Message_t;


Server Broadcast

The focus of server broadcast is how to calculate broadcast objects. Obviously, in a large map, a player's action on the eastmost side should not be visible to a player on the westmost side. So how can we calculate broadcast objects? The simplest way is to divide a map into small blocks and broadcast each time only a few blocks of players around it. So how much is appropriate? In general, the chunks are larger, the memory consumption increases, the chunks are smaller, and the CPU consumption increases (the reason will be mentioned later ). I personally think it is more appropriate to split a small block to the left and right of a screen. Each time nine small pieces of players around the broadcast, due to the frequent broadcast operations, the operations on the nine blocks around the bucket will become quite frequent. Therefore, if the block size is smaller, the range of the bucket will be expanded and the CPU resources will be quickly exhausted.

After cutting blocks, how can players move around between blocks? Let's think about what to do when switching a block. First, you need to figure out which of the nine players around the next block are not available in the current block, and broadcast your information to those players, at the same time, it is also necessary to determine which items in the nine blocks around the next block are not available now, and broadcast the information of those items to yourself, then, we can broadcast the information about the disappearance of the nine objects in the next block to ourselves, at the same time, it broadcasts the lost messages to those objects. This operation is not only cumbersome but also consumes a lot of CPU resources. How can we quickly calculate these items? Compare them one by one? Obviously, it seems that this is not a good solution. Here we can refer to some ideas of the Two-dimensional matrix collision detection to take nine blocks around ourselves as one matrix, and nine blocks around the target block as another matrix, check whether the two matrices collide. If the two matrices intersect, how can we calculate the blocks that do not overlap. Here, we can convert the coordinates of the intersection blocks into internal coordinates and then perform operations.

There is another solution for broadcasting. The implementation is not as simple as chunking. This method requires the client to assist in the computation. First, add a broadcast object queue in the connection structure of the server. When the client logs on to the server, the queue is sent to the client by the server, and the client maintains the queue by itself, when someone leaves the client's field of view, the client requests the server to send a message to the object that disappears. However, it is troublesome for someone to gain a general view.

First, each time the client sends an update position message to the server, the server calculates a field of view for the connection, and then loops the players on the entire map when the broadcast is required, find players whose coordinates are within their field of view. The advantage of using this method is that a large amount of messages need to be broadcast at one time when there is no conversion block. The disadvantage is that players on the whole map need to traverse when calculating the broadcast object, if there are too many players on a map, this operation will be slow.

Server Synchronization

Synchronization is very important in online games. It ensures that every player can see the same things on the screen. In fact, the simplest way to solve the synchronization problem is to broadcast the movements of each player to other players. There are actually two problems: 1. Broadcast to which players, which messages are broadcast. 2. What if network latency occurs. In fact, the first question is a very simple one. However, the reason why I raised this question is to remind everyone to consider this factor when designing their own message structure. The second problem is a very troublesome one. Let's take a look at this example:

For example, Player A sends a command to the server, saying that I am going to P2. The command is sent at T0, the server receives the command at T1, and then broadcasts the message to the players around it, the message content is "Player A from P1 to p2", there is a player B near a. The time for receiving this broadcast message from the server is T2. then, draw a picture on the client, A starts from P1 to P2. At this time there is a problem of non-synchronization, Players A and B on the screen shows a different picture of the T2-T1 time. What should we do at this time?

There is a solution. I name it "prediction pull". Although it is a little weird, basically everyone can understand it literally. To solve this problem, we must first define a value: prediction error. Then, you need to add an attribute in the connection class of each player on the server, called latency. Then, when the player logs in, the client time and server time are compared, the difference value is stored in latency. In the above example, when the server broadcasts a message, it calculates the currenttime of a client based on the latency of the object to be broadcast, and then contains the currenttime in the message header, then broadcast. In addition, a queue is created locally on the client of Player A to save the message. The message is deleted only in the message queue that has never been verified after server verification. If verification fails, it will be pulled back to Point P1. Then, when Player B receives the message "Player A from P1 to p2" sent from the server, it checks the server sending time in the message and compares it with the local time, if it is greater than the defined prediction error, even at the time of T2, Player A's screen goes to the location P3, and then the Player A on Player B's screen is pulled directly to P3, continue, so that synchronization can be ensured. Further, in order to ensure that the client runs more smooth, I do not recommend pulling the player directly, but calculate a bit of P4 after P3, and then use (P4-P1)/t (P4-P3) to calculate a very fast speed S, and then let player a quickly move to P4 with the speed S, this method is more reasonable, the prototype of this solution is known internationally as full plesiochronous. Of course, this prototype has been tampered with by me a lot to adapt to the synchronization of online games, so it becomes called prediction pull.

In another solution, I name it "verify synchronization" and I know the synchronization name. The general meaning is that each command is executed after it passes the server verification. The specific idea is as follows: first define a latency in each player connection type, and then the client will not move forward while responding to the player's mouse, instead, send a walking command to the server and wait for verification. After receiving the message, the server verifies the logic layer and calculates the range to be broadcast, including Player A. Different message headers are generated based on the latency of each client, start broadcasting. At this time, the gamer's walking information is completely synchronized. The advantage of this method is that it can ensure absolute synchronization between clients. The disadvantage is that when the network latency is large, the player's client behavior will become less smooth, it makes players feel uncomfortable. The prototype of this solution is known internationally as hierarchical master-slave synchronization. It has been widely used in various network fields since 1980s.

The last solution is an idealized solution, known internationally as mutual synchronization. It is a well-predicted solution for the future of the network. The reason for this solution is not to say that we have fully implemented this solution, but to apply some ideas of this solution to some aspects of the online game field. I named this solution semi-server synchronization. The general design concept is as follows:

First, the client needs to create many broadcast lists when logging on to the world. These lists are not synchronized in a timely manner on the backend of the client and the server. The reason for this is to create multiple lists, it is because there are more than one broadcast type, such as local message, remote message, and global message, these lists must be created based on the messages sent from the server during client login. When creating a list, you also need to obtain the latency of the broadcast object in each list, and maintain a complete user status list in the background, which is not synchronized with the server in time, according to the local user status table, some decisions can be decided by the client itself. When the client sends this decision, the final decision is directly sent to the client in each broadcast list, and the time is proofread to ensure that each client proofreads the received message based on the local time. Then, the method that uses the advance computing volume mentioned in the prediction pull to increase the speed of walking will make the synchronization very smooth. The advantage of this solution is that the synchronization between clients is not implemented through the server, which greatly reduces the errors caused by network latency, and most decisions can be made by the client, it also greatly reduces server resources. The drawback is that because both the message and decision-making power are placed on the client, the plug-in provides a great opportunity.
 

Contact Us

The content source of this page is from Internet, which doesn't represent Alibaba Cloud's opinion; products and services mentioned on that page don't have any relationship with Alibaba Cloud. If the content of the page makes you feel confusing, please write us an email, we will handle the problem within 5 days after receiving your email.

If you find any instances of plagiarism from the community, please send an email to: info-contact@alibabacloud.com and provide relevant evidence. A staff member will contact you within 5 working days.

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.