基于Netty从零开始搭建游戏服务器框架

一、Java环境配置

1.下载JDK

JDK8下载地址:https://www.oracle.com/java/technologies/javase/javase-jdk8-downloads.html

根据操作系统选择需要的相应版本即可

2.window 和 Linux 上的安装配置

2.1 window 环境

下载后JDK的安装根据提示进行,还有安装JDK的时候也会安装JRE,一并安装就可以了。
安装JDK,安装过程中可以自定义安装目录等信息,例如选择安装目录为 d:\Program Files (x86)\Java\jdk8

配置环境变量
1.安装完成后,右击"我的电脑",点击"属性",选择"高级系统设置";
2.选择"高级"选项卡,点击"环境变量";
3.在 "系统变量" 中新建2项属性,JAVA_HOME、CLASSPATH;
JAVA_HOME : D:\Program Files\Java\jdk8\jdk1.8.0_261
CLASSPATH : .;%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar;
4.在 "系统变量" 中设置 Path 变量,在里面添加
%JAVA_HOME%\bin;%JAVA_HOME%\jre\bin;
5.在cmd窗口输入 java -version 查看是否配置好,出现版本号表示配置成功

2.2 Linux 环境

1.创建一个 /user/local/java 目录,把下载的jdk-8u261-linux-x64.tar.gz包放入
2.解压jdk包,会解压到 jdk1.8.0_261 目录中
3.配置环境变量,可以在多个地方配置比如 ~/.bashrc,/etc/profile 等, 我这选择在/etc/profile添加

export JAVA_HOME=/usr/local/java/jdk1.8.0_231
export PATH=$PATH:$JAVA_HOME/bin
export CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar
export JRE_HOME=$JAVA_HOME/jre

4.执行 source /etc/profile
5.使用 java -version 查看是否配置好,出现版本号表示配置成功

二、游服框架搭建

1.开发环境

  1. 开发工具使用 IntelliJ IDEA 2019.2

  2. 项目使用 maven 管理,maven中央仓库(https://search.maven.org/) 需要的依赖包这里下载

  3. IDEA 需要安装 Maven Helper 插件,可以很方便执行相关maven命令

  4. 如果不使用IDEA中自带的maven工具,下载一个 apache-maven-3.6.1-bin.zip,下载url
    随便放置一个目录中解压出来即可,如目录 F:\Java\apache-maven-3.6.1-bin\apache-maven-3.6.1

  5. 配置 maven 环境
    5.1 设置环境变量
    增加一个 M2_HOME :F:\Java\apache-maven-3.6.1-bin\apache-maven-3.6.1
    在 Path 中添加 F:\Java\apache-maven-3.6.1-bin\apache-maven-3.6.1\bin;
    5.2 配置 conf/setting.xml
    a. 在 localRepository 项中可以设置自己的本地仓库路径,如 <localRepository>D:\tools\repository</localRepository>,也可使用默认,默认位置在 ${user.home}/.m2/repository
    b. 默认的中央仓库很慢有时候甚至连接不通,添加阿里云等镜像,添加镜像仓库的配置,在mirrors节点下面添加子节点:

    <!-- 阿里云仓库 -->
    <mirror>
        <id>alimaven</id>
        <mirrorOf>central</mirrorOf>
        <name>aliyun maven</name>
        <url>http://maven.aliyun.com/nexus/content/repositories/central/</url>
    </mirror>

    <!-- 中央仓库1 -->
    <mirror>
        <id>repo1</id>
        <mirrorOf>central</mirrorOf>
        <name>Human Readable Name for this Mirror.</name>
        <url>http://repo1.maven.org/maven2/</url>
    </mirror>

    <!-- 中央仓库2 -->
    <mirror>
        <id>repo2</id>
        <mirrorOf>central</mirrorOf>
        <name>Human Readable Name for this Mirror.</name>
        <url>http://repo2.maven.org/maven2/</url>
    </mirror>
c. <profiles>标签下添加一个<profile>标签,修改maven默认的JDK版本
    <profile>
      <id>jdk-1.8</id>
      <activation>
        <activeByDefault>true</activeByDefault>
        <jdk>1.8</jdk>
      </activation>
      <properties>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
        <maven.compiler.compilerVersion>1.8</maven.compiler.compilerVersion>
      </properties>
    </profile>
  1. IDEA下配置Maven
    File | Settings | Build, Execution, Deployment | Build Tools | Maven 下面设置
    a. Maven home directory 选择 刚刚解压的maven路径 F:\Java\apache-maven-3.6.1-bin\apache-maven-3.6.1
    b. 勾选Override,修改为自己目录下的settings.xml目录

2.搭建框架

1. 服务说明

游戏服务器一般来说是一组服务,如登陆服务(LoginServer),网关服务(GateServer),管理服务(GMServer),逻辑服务(LogicServer),DB服务(DBServer),地图服务(MapServer)以及一些其他业务服务等。

2.创建项目
1. 创建一个空项目,比如名称为 HpGameServer 的项目

File | New | Project 然后选择 Maven 来管理项目(用Gradle也行)|选择 Project SDK 选择安装好的jdk1.8即可 | Next | 填写GroupId(如:com.ali.hp) 和ArtifactId (HpGameServer)。把里面的代码文件夹删除留 pom.xml 文件管理整个项目

2. 公共子模块 HpCommon

在HpGameServer项目创建一个公共子模块 HpCommon 这个模块被其他服务共同使用,一些公用的包也可以都加在该模块下就不用每个服务都添加相同的依赖包了,其他服务只需添加 HpCommon 依赖就可以了。在HpGameServer下右键 New | Module 选择 Maven | roject SDK 选择自己安装的jdk1.8 | Next填写GroupId(com.ali.hp.HpCommon) 和ArtifactId (HpCommon)

在公共模块下的 pom.xml 添加一下公共依赖包,如netty包,protobuf包,log4j包等等

        <dependency>
            <groupId>com.google.protobuf</groupId>
            <artifactId>protobuf-java</artifactId>
            <version>3.4.0</version>
        </dependency>

        <dependency>
            <groupId>io.netty</groupId>
            <artifactId>netty-all</artifactId>
            <version>4.1.5.Final</version>
        </dependency>

        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.7.1</version>
        </dependency>
3. 网关服务 HpGateServer

在HpGameServer下右键 New | Module 选择 Maven | roject SDK 选择自己安装的jdk1.8 然后勾选 Create from archetype 选择 maven-archetype-quickstart 它会帮我们快速创建Maven工程模板 | Next填写GroupId(com.ali.hp.HpGateServer) 和ArtifactId (HpGateServer)
在网关服务的 pom.xml 下添加 HpCommon 包

    <dependency>
        <groupId>com.ali.hp.HpCommon</groupId>
        <artifactId>HpCommon</artifactId>
        <version>1.0.0</version>
    </dependency>

其他服务安装 HpGateServer 服务一样创建即可。

4. 试运行

在 HpGameServer 上右击 | Run Maven | install 全部编译完出现

INFO] HpGameServer ...................................... SUCCESS [  1.010 s]
[INFO] HpCommon ........................................... SUCCESS [  4.188 s]
[INFO] HpGateServer ....................................... SUCCESS [  8.816 s]
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time:  14.298 s

表示成功,说明依赖的包都下载OK了。

这时在 HpGateServer | target |HpGateServer-1.0.0.jar | 右键运行 会出现

 HpGameServer/HpGateServer/target/HpGateServer-1.0.0.jar中没有主清单属性

但是在 main 方法上可以运行,说明 jar 包没有找到 main 方法。

解决方案:
一、使用IDEA自带的构建jar包流程,不使用Maven打包
File | Project Structure... | Artifacts | + | JAR | From modules with dependencies | 从 All Modules 选择 HpGateServer | 点击 Main Class 自动找到各个子模块的 main 方法选择自己的,然后 Directory for MEAT-INF/MANIFEST.MF: 会自动填充路径:.../HpNettyServer/HpLoginServer/src/main/java 这里需要注意要把 /main/java去掉。

还有两个地方需要注意下

1.Output Directory 可以统一指定一个路径,这样所有的服务生成的jar包都在这个路径中了,比如指定在项目跟路径下的 out 目录下
2.勾选 Include in project build ,这样在IDEA工具栏上会生成相应的编译配置

测试:在 out 目录下生成的 HpGateServer.jar 右键运行 可以正常运行了。这样打jar包就可以部署到任何有java环境的地方运行啦。

3.代码编写

一. HpCommon

公共模块主要包括:

  1. 工具类,一些常用功能方法的集合,方便业务处理时调用。

  2. protobuf DSL文件和生成的文件,用于客户端与服务器通信的消息字段定义

  1. 自定义协议的编解码处理,自定义协议一般包括固定长度包头 + N字节的包体

    a. 客户端与服务器之间的包体稍微复杂一点,包体里面分两部分,一个验证头包和一个真正通信的消息包,验证头包确保数据的正确性和安全性

    b. 服务与服务之间的包体稍微简单一点,包体里面不用包括验证头包,但需要有个标记字段来标记是否为进行的protobuf封包

二. HpGateServer

使用netty编写服务端和客户端的代码模式大体比较固定这里举个例子说明,具体细节就不贴代码了,对应网关服务来说,既要编写服务端代码,又要编写客户端代码,它作为客户端与后端其他服务之间的桥梁,它相对其他后端服务来说它的角色就是一个客户端。

  1. 服务端代码
public class GateServer {

    private static final Logger log = LoggerFactory.getLogger(GateServer.class);
    private static final short listenPort = 8899;//网关监听端口
    private static final short logicPort = 1111;//逻辑服务端口
    private static final String logicIp = System.getProperty("logicIp"); //逻辑服务ip
    private static GateServer server = new GateServer();
    private Channel logicChannel = null;


    private GateServer(){}

    public static GateServer instance(){
        return server;
    }

    public void init() {

    }

    public void run() throws Exception {
        EventLoopGroup bossGroup = new NioEventLoopGroup(1);
        EventLoopGroup workerGroup = new NioEventLoopGroup(8);

        try {
            ServerBootstrap bootstrap = new ServerBootstrap();
            bootstrap.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .handler(new LoggingHandler(LogLevel.INFO))
                    .childHandler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel socketChannel) throws Exception {
                            ChannelPipeline pipeline = socketChannel.pipeline();
                            pipeline.addLast("pingpong", new IdleStateHandler(30, 0, 0, TimeUnit.SECONDS));
                            pipeline.addLast("msgDecoder", new MsgDecoder(65535, false));
                            pipeline.addLast("msgEncoder", new MsgEncoder(65536, false));
                            pipeline.addLast("gateServerHandler", new GateServerHandler());
                        }
                    })
                    .option(ChannelOption.SO_BACKLOG, 128)
                    .childOption(ChannelOption.SO_KEEPALIVE, true)
                    .childOption(ChannelOption.TCP_NODELAY, true);

            // Bind and start to accept incoming connections.
            ChannelFuture future = bootstrap.bind(listenPort).sync();
            System.out.println("linsten in "+ listenPort);
            // 连接后端
            new ConnBackend().connect(logicIp, logicPort);
            // Wait until the server socket is closed.
            future.channel().closeFuture().sync();
        } finally {
            workerGroup.shutdownGracefully();
            bossGroup.shutdownGracefully();
        }
    }

    public Channel getLogicChannel() {
        return logicChannel;
    }

    public void setLogicChannel(Channel channel) {
        logicChannel = channel;
    }

    public static short getLogicPort() {
        return logicPort;
    }

    public static String getLogicIp() {
        return logicIp;
    }

    public SocketAddress getLogicServerAddr() {
        return new InetSocketAddress(logicIp, logicPort);
    }

}
  1. 客户端代码
public class ConnBackend {

    private static final Logger log = LoggerFactory.getLogger(ConnBackend.class);
    private Channel backendChannel = null;


    public Channel getBackendChannel() {
        return backendChannel;
    }

    public void connect(String ip, short port) {
        EventLoopGroup group = new NioEventLoopGroup(1);
        Bootstrap bootstrap = new Bootstrap();

        try {
            bootstrap.group(group);
            bootstrap.channel(NioSocketChannel.class);
            bootstrap.option(ChannelOption.TCP_NODELAY, true);
            bootstrap.option(ChannelOption.SO_KEEPALIVE, true);
            bootstrap.handler(new ChannelInitializer<Channel>() {
                @Override
                protected void initChannel(Channel ch) {
                    ChannelPipeline pipeline = ch.pipeline();
                    pipeline.addLast("pingpong", new IdleStateHandler(180, 30, 0, TimeUnit.SECONDS));
                    pipeline.addLast("msgDecoder", new MsgDecoder(65536, true));
                    pipeline.addLast("msgEncoder", new MsgEncoder(65536, true));
                    pipeline.addLast("connBackendHandler", new ConnBackendHandler());
                }
            });
            ChannelFuture future = bootstrap.connect(new InetSocketAddress(ip, port)).
                    addListener(new ConnBackendListener()).sync();
            backendChannel = future.channel();
            System.out.println("backendChannel:" + backendChannel);
            future.channel().closeFuture().sync();
        } catch (Exception e) {
            //e.printStackTrace();
            System.err.println("连接后端:[" + ip +":"+ port + "]出错,检查后端是否开启!!!");
        }
    }

}

客户端代码需要注意断开重连的处理,addListener(new ConnBackendListener())添加一个对该通道的监听,断开连接后需要在监听器里再次发起连接

三、可执行程序的部署和运行

1.window 环境

一般来说 window 上运行只是测试而已
1.可以创建一个目录 如 /f/game_server, 把生成的各服务的 jar (比如网关服务 HpGateServer.jar)拷贝到该目录
2.在该目录里创建一个 start.bat 文件,内容如下

START "HpGateServer" javaw -Dip=192.168.121.170 -Dxxx -jar HpGateServer.jar  //这个是放在后台运行命令,-D后面是需要的参数,可以根据实际需要填写
rem java -Dip=192.168.121.170 -Dxxx  -jar HpNettyClient.jar // rem是注释,这个命令不会放在后台执行

pause

3.停止在后台运行的程序,在任务管理器中 结束掉 java、javaw 进程即可

2.Linux 环境

  1. 创建放jar包的目录 比如/home/game_server,多组服务可以创建相应的字目录 如: gate_server, login_server 等
  2. 创建 start.sh 脚本启动服务,内容如下
#!/bin/sh
echo "start game_server======================================================"  

echo "start gate_server"                                        
cd ./gate_server                                        
nohup java -jar HpGateServer.jar  > /dev/null 2>&1 &            
cd ../                                                  

echo "start login_server"                                       
cd ./login_server                                       
nohup java -jar HpLoginServer.jar  > /dev/null 2>&1 &           
cd ../                                                  

ps -ef|grep Hp|grep -v grep

3 . 创建 stop.sh 脚本停止服务,内容如

echo "stop gate_server"
pid=`ps -ef|grep HpGateServer|grep -v _SH|grep -v grep|awk '{print $2}'`
if [ "$pid" == "" ]; then
    echo "gate_server not start"
else
    kill -9 $pid
fi

echo "stop login_server"
pid=`ps -ef|grep HpLoginServer|grep -v grep|awk '{print $2}'`
if [ "$pid" == "" ]; then
    echo "login_server not start"
else
    kill -9 $pid
fi

echo "kill all server OK"

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351