【Zinx应用-MMO游戏案例-(6)世界聊天】Golang轻量级并发服务器框架

Zinx源代码

github
https://github.com/aceld/zinx
gitee码云
https://gitee.com/Aceld/zinx


在线开发教程

【B站】
zinx视频教程-Golang轻量级TCP服务器框架-适合自学者

【YouTube】
zinx开发YouTube中国版


【Zinx教程目录】
完整教程电子版(在线高清)-下载
Zinx框架视频教程(框架篇)(完整版下载)链接在下面正文
Zinx框架视频教程(应用篇)(完整版下载)链接在下面正文
Zinx开发API文档
Zinx第一章-引言
Zinx第二章-初识Zinx框架
Zinx第三章-基础路由模块
Zinx第四章-全局配置
Zinx第五章-消息封装
Zinx第六章-多路由模式
Zinx第七章-读写分离模型
Zinx第八章-消息队列及多任务
Zinx第九章-链接管理
Zinx第十章-连接属性设置


【Zinx应用案例-MMO多人在线游戏】
(1)案例介绍
(2)AOI兴趣点算法
(3)数据传输协议protocol buffer
(4)Proto3协议定义
(5)构建项目及用户上线
(6)世界聊天
(7)上线位置信息同步
(8)移动位置与AOI广播
(9)玩家下线
(10)模拟客户端AI模块


七、世界聊天系统实现

7.1 世界管理模块

​ 现在需要一个管理当前世界所有玩家的一个管理器,管理器应该拥有全部的当前在线玩家信息和当前世界的AOI划分规则。方便玩家与玩家之间进行聊天,同步位置等功能。

​ 首先,在创建一个world_manager.go文件作为世界管理器模块。

mmo_game/core/world_manager.go

package core

import (
    "sync"
)

/*
    当前游戏世界的总管理模块
*/
type WorldManager struct {
    AoiMgr  *AOIManager       //当前世界地图的AOI规划管理器
    Players map[int32]*Player //当前在线的玩家集合
    pLock   sync.RWMutex      //保护Players的互斥读写机制
}

//提供一个对外的世界管理模块句柄
var WorldMgrObj *WorldManager

//提供WorldManager 初始化方法
func init() {
    WorldMgrObj = &WorldManager{
        Players: make(map[int32]*Player),
        AoiMgr:  NewAOIManager(AOI_MIN_X, AOI_MAX_X, AOI_CNTS_X, AOI_MIN_Y, AOI_MAX_Y, AOI_CNTS_Y),
    }
}

//提供添加一个玩家的的功能,将玩家添加进玩家信息表Players
func (wm *WorldManager) AddPlayer(player *Player) {
    //将player添加到 世界管理器中
    wm.pLock.Lock()
    wm.Players[player.Pid] = player
    wm.pLock.Unlock()

    //将player 添加到AOI网络规划中
    wm.AoiMgr.AddToGridByPos(int(player.Pid), player.X, player.Z)
}

//从玩家信息表中移除一个玩家
func (wm *WorldManager) RemovePlayerByPid(pid int32) {
    wm.pLock.Lock()
    delete(wm.Players, pid)
    wm.pLock.Unlock()
}

//通过玩家ID 获取对应玩家信息
func (wm *WorldManager) GetPlayerByPid(pid int32) *Player {
    wm.pLock.RLock()
    defer wm.pLock.RUnlock()

    return wm.Players[pid]
}

//获取所有玩家的信息
func (wm *WorldManager) GetAllPlayers() []*Player {
    wm.pLock.RLock()
    defer wm.pLock.RUnlock()

    //创建返回的player集合切片
    players := make([]*Player, 0)

    //添加切片
    for _, v := range wm.Players {
        players = append(players, v)
    }

    //返回
    return players
}

​ 该模块主要是将AOI和玩家做了一层统一管理,起到协调其他模块的中间功能。其中有一个全局变量WorldMgrObj是对外开放的管理模块句柄。供其他模块使用。

​ 现在我们应该在玩家上线的时候,也将玩家添加到WorldMgrObj中。

mmo_game/server.go

//当客户端建立连接的时候的hook函数
func OnConnecionAdd(conn ziface.IConnection)  {
    //创建一个玩家
    player := core.NewPlayer(conn)
    //同步当前的PlayerID给客户端, 走MsgID:1 消息
    player.SyncPid()
    //同步当前玩家的初始化坐标信息给客户端,走MsgID:200消息
    player.BroadCastStartPosition()
    
    //========将当前新上线玩家添加到worldManager中
    core.WorldMgrObj.AddPlayer(player)
    //========================================

    fmt.Println("=====> Player pidId = ", player.Pid, " arrived ====")
}

7.2 世界聊天系统实现

​ 接下来,我们来做一个玩家和玩家之间的世界聊天广播功能。

【Zinx应用-MMO游戏案例-(6)世界聊天】Golang轻量级并发服务器框架_第1张图片
17-zinx游戏案例-世界聊天流程.png

A) proto3协议定义

​ 这里涉及到了MsgId:2的指令,还有对应的Talk的proto协议。

MsgID:2

Talk:

  • 同步玩家本次登录的ID(用来标识玩家), 玩家登陆之后,由Server端主动生成玩家ID发送给客户端
  • 发起者: Client
  • Content: 聊天信息
message Talk{
    string Content=1;
}

所以我们应该先修改proto文件

mmo_game/pb/msg.proto

syntax="proto3";                //Proto协议
package pb;                     //当前包名
option csharp_namespace="Pb";   //给C#提供的选项

//同步客户端玩家ID
message SyncPid{
    int32 Pid=1;
}

//玩家位置
message Position{
    float X=1;
    float Y=2;
    float Z=3;
    float V=4;
}

//玩家广播数据
message BroadCast{
    int32 Pid=1;
    int32 Tp=2;              //1-世界聊天  2-玩家位置
    oneof Data {
        string Content=3;    //聊天的信息
        Position P=4;        //广播用户的位置
        int32 ActionData=5;
        }
}

//=====================
//玩家聊天数据
message Talk{
    string Content=1;    //聊天内容
}
//=====================

执行build.sh 生成新的msg.proto.go文件。

B) 聊天业务API建立

​ 接下来,我们创建一个api文件

mmo_game/api/world_chat.go

package api

import (
    "fmt"
    "github.com/golang/protobuf/proto"
    "zinx/ziface"
    "zinx/zinx_app_demo/mmo_game/core"
    "zinx/zinx_app_demo/mmo_game/pb"
    "zinx/znet"
)

//世界聊天 路由业务
type WorldChatApi struct {
    znet.BaseRouter
}

func (*WorldChatApi) Handle(request ziface.IRequest) {
    //1. 将客户端传来的proto协议解码
    msg := &pb.Talk{}
    err := proto.Unmarshal(request.GetData(), msg)
    if err != nil {
        fmt.Println("Talk Unmarshal error ", err)
        return
    }

    //2. 得知当前的消息是从哪个玩家传递来的,从连接属性pid中获取
    pid, err := request.GetConnection().GetProperty("pid")
    if err != nil {
        fmt.Println("GetProperty pid error", err)
        request.GetConnection().Stop()
        return
    }
    //3. 根据pid得到player对象
    player := core.WorldMgrObj.GetPlayerByPid(pid.(int32))

    //4. 让player对象发起聊天广播请求
    player.Talk(msg.Content)
}

​ 这里实际上对于msgID:2的路由业务函数的实现。其中有个小细节需要注意一下。第2步,根据链接conn得到当前玩家的pid,应该是我们之前在玩家上线的时候,将pid和conn做一个属性绑定,如下:

mmo_game/server.go

//当客户端建立连接的时候的hook函数
func OnConnecionAdd(conn ziface.IConnection)  {
    //创建一个玩家
    player := core.NewPlayer(conn)
    //同步当前的PlayerID给客户端, 走MsgID:1 消息
    player.SyncPid()
    //同步当前玩家的初始化坐标信息给客户端,走MsgID:200消息
    player.BroadCastStartPosition()
    //将当前新上线玩家添加到worldManager中
    core.WorldMgrObj.AddPlayer(player)
    //=================将该连接绑定属性Pid===============
    conn.SetProperty("pid", player.Pid)
    //===============================================
    fmt.Println("=====> Player pidId = ", player.Pid, " arrived ====")
}

接下来,我们来看一下Player里的Talk实现方法:

mmo_game/core/player.go

//广播玩家聊天
func (p *Player) Talk(content string) {
    //1. 组建MsgId200 proto数据
    msg := &pb.BroadCast{
        Pid:p.Pid,
        Tp:1,//TP 1 代表聊天广播
        Data: &pb.BroadCast_Content{
            Content: content,
        },
    }

    //2. 得到当前世界所有的在线玩家
    players := WorldMgrObj.GetAllPlayers()

    //3. 向所有的玩家发送MsgId:200消息
    for _, player := range players {
        player.SendMsg(200, msg)
    }
}

C) 测试世界聊天功能

我们在服务端运行server

$go run server.go
$ go run server.go 
Add api msgId =  2
[START] Server name: Zinx Game,listenner at IP: 0.0.0.0, Port 8999 is starting
[Zinx] Version: V0.11, MaxConn: 3000, MaxPacketSize: 4096
start Zinx server   Zinx Game  succ, now listenning...
Worker ID =  9  is started.
Worker ID =  4  is started.
Worker ID =  5  is started.
Worker ID =  6  is started.
Worker ID =  7  is started.
Worker ID =  8  is started.
Worker ID =  0  is started.
Worker ID =  1  is started.
Worker ID =  2  is started.
Worker ID =  3  is started.

打开两个客户端,分别互相聊天,效果如下,我们的聊天功能已经实现了。

【Zinx应用-MMO游戏案例-(6)世界聊天】Golang轻量级并发服务器框架_第2张图片
18-Zinx游戏案例-聊天场景1.png
【Zinx应用-MMO游戏案例-(6)世界聊天】Golang轻量级并发服务器框架_第3张图片
19-Zinx游戏案例-聊天场景2.png

关于作者:

作者:Aceld(刘丹冰)
号:IT无崖子

mail: [email protected]
github: https://github.com/aceld
原创书籍gitbook: http://legacy.gitbook.com/@aceld

原创声明:未经作者允许请勿转载, 如果转载请注明出处

你可能感兴趣的:(【Zinx应用-MMO游戏案例-(6)世界聊天】Golang轻量级并发服务器框架)