美文网首页
游戏服务器中多线程之间如何通信

游戏服务器中多线程之间如何通信

作者: 王广帅 | 来源:发表于2018-12-26 15:28 被阅读15次

在游戏服务器中,线程的管理是重中之重,在上一篇文章中,已经简单说明了,在游戏服务器开多少线程合适的问题,点击这里查看,因为线程数量不能太多,所以为了提高游戏服务器并发性,就需要在线程处理业务的速度要快,不能长时间卡住线程,比如,不能有网络io,磁盘IO等耗时的操作。所以我们会把有限的线程数进配按需分配。线程的主要分配方式如下:

1,与客户端的IO线程

负责接收客户端消息,和向客户端发送消息。

2,处理玩家业务的逻辑线程

负责业务逻辑的处理与计算。

3,处理rpc或数据库同步的网络线程。

负责不同服务之间的通信

4,处理日志的磁盘IO 线程

相应的线程分别做自己该做的时间。但是这时候问题来了,既然按用途分配了这些线程,那不可免费的就会出现不同线程之间的数据交互了。举个例子来说,比如玩家登陆和排队,假如说一个区一台物理机,处理与客户端的IO线程占1个,业务逻辑线程16个,请求数据库的线程3个。其它的先忽略不算。这个时候,服务器收到客户端登陆的请求,业务线程收到这个请求,第一件事情是去数据库查用户的信息,因为业务逻辑线程是多个用户之间共用的,所以你不能在这里等待数据库请求的返回,要不然会卡别的业务逻辑处理。最好的做法就是,把这个请求封装成一个事件,发送到数据库同步线程中去处理这个事。等待数据库线程处理完了,再把结果告诉业务逻辑线程,然后再处理业务,这个时候处理的数据都在内存中了,所以速度非常快,处理完之后再返回客户端。

有人说,那这样还不是卡登陆吗?是的,必卡登陆,因为处理数据库的线程就那么3个。对于mysql的查询性能测试如下:

我们3个线程按1500个计算,那么也只有登陆并发达到1500的时候,才会觉得服务慢一些,(实际情况可能会再底一些),如果并发能达到这么高,说明游戏很火了,可以等待数钱了。如果再多的人,我们就可以使用排队的功能的。可以查询到数据库线程中正在等待执行任务的数量,如果达到某个值,就可以在业务服务中给客户端返回正在排队的,让客户端过一会再来请求即可。

回到正题,那么游戏服务器线程之间该如何传递数据呢,可能每个人的做法不一样,这里只列举中一个,希望给大家给带来些参考,就当是交流学习了。如果你有更好的方法,也希望您评论分享。

在Java中,多线程之间交互数据,即A线程给B线程一个请求事件,A线程还要获取B线程的执行结果,比如登陆,请数据库线程去查库,查完之后告诉逻辑线程。Java提供了一个Future/Callable的机制,详细大家可以自动百度它们的用法,但是它们有一个缺点,就是A线程在获取结果时(调用future.get())的时候是同步的,如果B线程没有执行完,还是会卡A线程。这显示不是我们想要的。对于这一点,我们伟大的异步框架,netty的作者给出了一个解决方法,那就是Future / promise模式。这里对它的源码暂不做分皙了,有时间另写文章说明吧,有兴趣的同学可以自己查看。这里只说一下它的用法。

此代码只是模拟,真实应用中还需要自己设计,首先是数据库查询的管理类:

package com.xinyue.demo.future;

import io.netty.util.concurrent.DefaultEventExecutor;

import io.netty.util.concurrent.Future;

import io.netty.util.concurrent.Promise;

public class MysqlManager {

    DefaultEventExecutor threadB = new DefaultEventExecutor();

    // 获取当前任务数量

    public int getTaskCount() {

        return threadB.pendingTasks();

    }

    public Future<User> queryUser(String name, String pasword, Promise<User> promise) {

        threadB.execute(() -> {

            // 这里在B线程中执行果查询数据的操作

            User user = null;// 这里由查询结果赋值

            promise.setSuccess(user);

        });

        return promise;

    }

}

然后是业务管理类:

package com.xinyue.demo.future;

import io.netty.channel.ChannelHandlerContext;

import io.netty.util.concurrent.DefaultEventExecutor;

import io.netty.util.concurrent.DefaultPromise;

import io.netty.util.concurrent.EventExecutor;

import io.netty.util.concurrent.Future;

import io.netty.util.concurrent.GenericFutureListener;

import io.netty.util.concurrent.Promise;

public class LogicManager {

    EventExecutor threadA = new DefaultEventExecutor();

    private MysqlManager mysqlManager;

    public LogicManager(MysqlManager mysqlManager) {

        this.mysqlManager = mysqlManager;

    }

    /**

    * 这里模拟的就是业务线程收到登陆的请求之后,然后业务线程要查数据库,但是查询数据库的操作是在另外一个线程中执行的,查询结果出来之后,再返回结果。

    * @param name

    * @param password

    * @param ctx

    */

    public void login(String name,String password,ChannelHandlerContext ctx) {

        //这里创建Promise的时候,传入线程A,表示这个promise中的任何操作都会在线程A中发生。

        Promise<User> promise = new DefaultPromise<>(threadA);

        //这里面是一个异步调用,所以不会卡当前的业务线程A.

        Future<User> future = mysqlManager.queryUser(name, password, promise);

        future.addListener(new GenericFutureListener<Future<User>>() {

            @Override

            public void operationComplete(Future<User> future) throws Exception {

                //这里会在promise调用setSuccess的时候时候执行,而且这个监听方法的执行是在A线程中。

                User user =future.get();

                //这里就拿到了用户的信息了,然后可以做一系列的业务操作,完成之后,再返回给客户端消息。

                ctx.writeAndFlush("登陆成功");

            }

        });

    }

}

名仕爵情侣手表一对心形1314韩版潮流学生防水石英表免费刻字男女

 通过这个例子,其它关于多线程交互数据的功能实现,大家就可以举一返回三了。

 欢迎加群交流,QQ群:66728073,197321069,398808948 

相关文章

  • 游戏服务器中多线程之间如何通信

    在游戏服务器中,线程的管理是重中之重,在上一篇文章中,已经简单说明了,在游戏服务器开多少线程合适的问题,点击这里查...

  • JAVA多线程(五)

    这张我们讲讲多线程之间通信和ThreadGroup! 单线程间通信: 如果服务器端有若干个线程会从队列中获取想要的...

  • go并发编程之美(一)

    一、前言 在Java中多线程之间是通过共享内存进行通信的,在go中多线程之间通信是基于消息的,go中的通道是go中...

  • Java 多线程(三)优化任务执行

    本篇文章通过服务器通信和页面渲染两个功能的实现来加深多线程中Future和Executor的理解。 服务器通信 串...

  • 游戏服务器启动多少个线程合适

    线程是服务器的一种希有资源,它的创建,销毁,切换都需要很多服务器的其它资源。而在游戏服务器中,只要没有在多线程之间...

  • 2018-08-08 多线程服务器

    使用多线程实现多个客户端与服务器通信

  • 1.为什么要使用网络?

    客户端和服务器之间如何通信? 简而言之:客户端和服务器之间通信,由于使用不同语言编程,所以使用HTTP协议,完成双...

  • 最详细分析Java 内存模型

    并发编程中, 线程之间如何通信及线程之间如何同步, 通信是指线程之间以何种机制来交换信息。在命令式编程中,线程之间...

  • 牛客网Java练习6

    1.在一个基于分布式的游戏服务器系统中,不同的服务器之间,哪种通信方式是不可行的()? A 管道B 消息队列C 高...

  • 第四章 并发编程基础

    线程的状态。如何启动和终止线程。多线程之间通信的基本方式。多线程编程的等待/通知的经典范式。 现代操作系统调度的最...

网友评论

      本文标题:游戏服务器中多线程之间如何通信

      本文链接:https://www.haomeiwen.com/subject/szuhlqtx.html