【Zinx应用-MMO游戏案例-(8)移动位置与AOI广播】Golang轻量级并发服务器框架

Zinx源代码

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


在线开发教程

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

【YouTube】
zinx开发YouTube中国版

微信端文档

【Zinx应用-MMO游戏案例-(8)移动位置与AOI广播】Golang轻量级并发服务器框架_第1张图片
技术资源分享.jpg

【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模块


现在我们来添加玩家移动的时候,周边玩家显示同步位置,具体流程图,如下:
【Zinx应用-MMO游戏案例-(8)移动位置与AOI广播】Golang轻量级并发服务器框架_第2张图片
23-Zinx游戏案例-移动位置未跨越格子流程图.png

这里面涉及到两个消息MsgID:3和 MsgID200,Tp=4。当玩家移动的时候,客户端会主动给服务端发送MsgID:3的消息. 所以首先,我们应该给服务端注册MsgID:3的路由处理业务

mmo_game/ server.go

func main() {
    //创建服务器句柄
    s := znet.NewServer()

    //注册客户端连接建立和丢失函数
    s.SetOnConnStart(OnConnecionAdd)

    //注册路由
    s.AddRouter(2, &api.WorldChatApi{})    //聊天
    s.AddRouter(3, &api.MoveApi{})         //移动

    //启动服务
    s.Serve()
}

接下来,我们需要创建一个api接口,实现MoveApi{}模块.

mmo_game/api/move.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 MoveApi struct {
    znet.BaseRouter
}


func (*MoveApi) Handle(request ziface.IRequest) {
    //1. 将客户端传来的proto协议解码
    msg := &pb.Position{}
    err := proto.Unmarshal(request.GetData(), msg)
    if err != nil {
        fmt.Println("Move: Position 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
    }

    fmt.Printf("user pid = %d , move(%f,%f,%f,%f)", pid, msg.X, msg.Y, msg.Z, msg.V)

    //3. 根据pid得到player对象
    player := core.WorldMgrObj.GetPlayerByPid(pid.(int32))

    //4. 让player对象发起移动位置信息广播
    player.UpdatePos(msg.X, msg.Y, msg.Z, msg.V)
}

move.go的业务和我们之前的world_chat.go的业务很像。最后调用了Player.UpdatPos()方法,该方法是主要处理及发送同步消息的方法。我们接下来一起实现这个方法.

mmo_game/core/player.go

//广播玩家位置移动
func (p *Player) UpdatePos(x float32, y float32, z float32, v float32) {
    //更新玩家的位置信息
    p.X = x
    p.Y = y
    p.Z = z
    p.V = v

    //组装protobuf协议,发送位置给周围玩家
    msg := &pb.BroadCast{
        Pid:p.Pid,
        Tp:4,     //4 - 移动之后的坐标信息
        Data: &pb.BroadCast_P{
            P:&pb.Position{
                X:p.X,
                Y:p.Y,
                Z:p.Z,
                V:p.V,
            },
        },
    }

    //获取当前玩家周边全部玩家
    players := p.GetSurroundingPlayers()
    //向周边的每个玩家发送MsgID:200消息,移动位置更新消息
    for _, player := range players {
        player.SendMsg(200, msg)
    }
}

//获得当前玩家的AOI周边玩家信息
func (p *Player) GetSurroundingPlayers() []*Player {
    //得到当前AOI区域的所有pid
    pids := WorldMgrObj.AoiMgr.GetPidsByPos(p.X, p.Z)

    //将所有pid对应的Player放到Player切片中
    players := make([]*Player, 0, len(pids))
    for _, pid := range pids {
        players = append(players, WorldMgrObj.GetPlayerByPid(int32(pid)))
    }

    return players
}

​ 其中GetSurroundingPlayers()是获取当前玩家AOI周边的玩家Player对象有哪些。

该方法的整体思路是获取周边的所有玩家,发送位置更新信息。

下面我们再次启动服务器,同时开3个客户端,看看最后的效果。

24-Zinx游戏案例-位置信息移动效果.gif

显示证明,3个客户端已经可以实现移动同步的过程,那么实际上,我们基本的MMO大型网游在线游戏的基础模型已经搭建完成了,接下来至于添加一些其他的游戏机制,比如对战,积分等。实际上可以基于这个开发架构和流程继续迭代开发了。


关于作者:

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

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

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

你可能感兴趣的:(【Zinx应用-MMO游戏案例-(8)移动位置与AOI广播】Golang轻量级并发服务器框架)