在系列[1]中,我们展示了RPC调用协议的定义以及演示,通过方法定义以及协议约定,进行了协议约定以及调用过程的约定。
然而,实际上在游戏中,调用过程之后,需要传输相对多的数据给服务端。

常用场景,客户端使用金币购买一把木剑。

一般情况下我们会这么约定:

              /// <summary>
/// 购买的返回消息
/// </summary>
public class ByItemReturn
{
/// <summary>
/// 购买的物品ID
/// </summary>
public int ItemId { get; set; } /// <summary>
/// 购买的物品数量
/// </summary>
public int Count { get; set; } /// <summary>
/// 玩家剩余金钱/或减少金钱
/// </summary>
public int NowGold { get; set; } }

  

方法定义和实现:

            /// <summary>
/// 购买物品
/// </summary>
/// <param name="itemId">物品id</param>
/// <returns></returns>
public static ByItemReturn BuyItemByGold(int itemId)
{
//伪代码
//获取玩家信息
//查找购买价格
//检查玩家金币
//玩家金币减少
//获取玩家背包
//添加背包物品
return new ByItemReturn()
{
Count = 1,
Code = 0,
ItemId = 100,
NowGold = 900
};
}

  

客户端调用:

            /// <summary>
/// 客户端方法
/// </summary>
/// <param name="itemId"></param>
public static void DoBuyItemByGold(int itemId)
{
//执行调用
BuyItemByGold(itemId, (r) =>
{
//显示购买成功
//把金币付给本地数据
//刷新本地金币
});
}

  

等等,假如我们修改了需求,我们允许拿钻石购买木剑。

修改代码:

            /// <summary>
/// 购买的返回消息
/// </summary>
public class ByItemReturn2
{
/// <summary>
/// 购买的物品ID
/// </summary>
public int ItemId { get; set; } /// <summary>
/// 购买的物品数量
/// </summary>
public int Count { get; set; } /// <summary>
/// 玩家剩余金钱/或减少金钱
/// </summary>
public int NowGold { get; set; } /// <summary>
/// 玩家剩余钻石
/// </summary>
public int NowDiamond { get; set; } } /// <summary>
/// 购买物品
/// </summary>
/// <param name="itemId">物品id</param>
/// <returns></returns>
public static ByItemReturn2 BuyItemByGold2(int itemId,bool diamond=false, Action<ByItemReturn2> callback = null)
{
//伪代码
//获取玩家信息
//查找购买价格
//检查玩家金币
//玩家金币减少
//检查玩家钻石
//玩家钻石减少
//获取玩家背包
//添加背包物品
return new ByItemReturn2()
{
Count = 1,
Code = 0,
ItemId = 100,
NowGold = 900,
NowDiamond = 100
};
} /// <summary>
/// 客户端方法
/// </summary>
/// <param name="itemId"></param>
public static void DoBuyItemByGold2(int itemId, bool diamond = false)
{
//执行调用
BuyItemByGold2(itemId,diamond, (r) =>
{
//显示购买成功
//把金币付给本地数据
//刷新本地金币
//刷新本地钻石
});
}

  

假设我们有个通用协议,能描述所有资源的修改,就能轻松的搞定这个问题了。
假如我们代码变成这样:

	            /// <summary>
/// 购买的返回消息
/// </summary>
public class ByItemReturn
{
/// <summary>
/// 购买的物品ID
/// </summary>
public int ItemId { get; set; } /// <summary>
/// 购买的物品数量
/// </summary>
public int Count { get; set; } /// <summary>
/// 修改信息
/// </summary>
public object ChangeMessage { get; set; }
} /// <summary>
/// 购买物品
/// </summary>
/// <param name="itemId">物品id</param>
/// <returns></returns>
public static ByItemReturn BuyItemByGold(int itemId,Action<ByItemReturn> callback=null)
{
//伪代码
//获取玩家信息
//查找购买价格
//检查玩家金币
//玩家金币减少
//获取玩家背包
//添加背包物品
return new ByItemReturn()
{
Count = 1,
ItemId = 100,
ChangeMessage = GetChangeMessage()
};
} public static object GetChangeMessage()
{
return null;
}
public static object SetChangeMessage(object message)
{
return null;
} /// <summary>
/// 客户端方法
/// </summary>
/// <param name="itemId"></param>
public static void DoBuyItemByGold(int itemId)
{
//执行调用
BuyItemByGold(itemId, (r) =>
{
//显示购买成功
//设置修改数据
SetChangeMessage(r.ChangeMessage);
//刷新显示面板
});
}

  

至少,看起来代码变短了。
遇到之前的情况,协议不用修改,如果有多个类似的接口,相对修改也是减少了。
基本修改逻辑,调整接口,调整调用。 那么也就重新完成需求了。

都写个GetChangeMessage SetChangeMessage 多麻烦,我们拿AOP搞定吧 会怎么样。
假设我们的所有callback之前都有个SetChangeMessage
所有的接口返回之前都会自动调用下GetChangeMessage
看看新代码

    /// <summary>
/// 购买的返回消息
/// </summary>
public class ByItemReturn
{
/// <summary>
/// 购买的物品ID
/// </summary>
public int ItemId { get; set; } /// <summary>
/// 购买的物品数量
/// </summary>
public int Count { get; set; } /// <summary>
/// 修改信息
/// </summary>
public object ChangeMessage { get; set; }
} /// <summary>
/// 购买物品
/// </summary>
/// <param name="itemId">物品id</param>
/// <returns></returns>
public static ByItemReturn BuyItemByGold(int itemId,Action<ByItemReturn> callback=null)
{
//伪代码
//获取玩家信息
//查找购买价格
//检查玩家金币
//玩家金币减少
//获取玩家背包
//添加背包物品
return new ByItemReturn()
{
Count = 1,
ItemId = 100,
};
} /// <summary>
/// 客户端方法
/// </summary>
/// <param name="itemId"></param>
public static void DoBuyItemByGold(int itemId)
{
//执行调用
BuyItemByGold(itemId, (r) =>
{
//显示购买成功
//刷新显示面板
});
}

  

似乎好像更好了。
那么问题来了,这样的一个想法不错,怎么实现呢?
我怎么知道哪个数据是否修改?
回头看看系列文章1的响应定义: 差异数据 就是所谓的 ChangeMessage

   /// <summary>
/// 返回数据
/// </summary>
public partial class ResponseObj
{
/// <summary>
/// 返回数据对象
/// </summary>
public object result { get; set; } /// <summary>
/// 是否出错信息
/// 默认0 无出错信息
/// </summary>
public int error { get; set; } /// <summary>
/// 请求序号
/// </summary>
public int cid { get; set; } /// <summary>
/// 差异数据
/// </summary>
public OpChangeItem[] opstr{ get; set; } }

  

协议部分后续如果有同学感兴趣,到时候在开一篇,现在可以暂且无视。
首先,我们的假定:
1.我们总是有办法监测到是否有一个字段被修改了。
2.我们总是可以知道,通过修改的值设置到具体某个实例里面的某个字段

也就是可以得到这样的一个类似描述。

首先客户端和服务端都有一份这个数据:

{
"钻石": 100,
"金币": 100,
"背包": [
{
"ID": 101,
"耐久": 10
}
]
}

  

我们有个方法,叫木剑耐久降低1

通过以上的假设,我们可以得到一句修改描述

ID为101的木剑中的耐久变成9.

那么我们可以把这句话还原,找到101这把木剑,然后把耐久改成9.

{
"钻石": 100,
"金币": 100,
"背包": [
{
"ID": 101,
"耐久": 10 -> 9
}
]
}

  

客户端,的数据就和服务端是一样的,然后再执行回调方法,看看我们得到了什么。
我们直接可以从本地数据来获得这些信息,而不用服务端传递给客户端。

然后这个事情就变得更简单了。
重复上面的例子。

            /// <summary>
/// 购买物品
/// </summary>
/// <param name="itemId">物品id</param>
/// <returns>返回物品ID</returns>
public static int BuyItemByGold(int itemId,Action<ByItemReturn> callback=null)
{
//伪代码
//获取玩家信息
//查找购买价格
//检查玩家金币
//玩家金币减少
//获取玩家背包
//添加背包物品
return newId;
} /// <summary>
/// 客户端方法
/// </summary>
/// <param name="itemId"></param>
public static void DoBuyItemByGold(int itemId)
{
//执行调用
BuyItemByGold(itemId, (r) =>
{
//显示购买成功
//从背包中查找那个ID=r的对象
//展示这个道具
//刷新显示面板
});
}

  

似乎好像更精简了,有没有。而实际上你可能获取的数量不一定为1,所以返回值还是跑不掉。
但是在很多时候确实可以达到这样的效果。
等等,是不是漏了点什么,如果这样可以达到开发的目的,我还要服务端干嘛?

游戏编程系列[2]--游戏编程中RPC与OpLog协议的结合--序的更多相关文章

  1. WCF编程系列(六)以编程方式配置终结点

    WCF编程系列(六)以编程方式配置终结点   示例一中我们的宿主程序非常简单:只是简单的实例化了一个ServiceHost对象,然后调用open方法来启动服务.而关于终结点的配置我们都是通过配置文件来 ...

  2. Java并发编程系列-(1) 并发编程基础

    1.并发编程基础 1.1 基本概念 CPU核心与线程数关系 Java中通过多线程的手段来实现并发,对于单处理器机器上来讲,宏观上的多线程并行执行是通过CPU的调度来实现的,微观上CPU在某个时刻只会运 ...

  3. java并发编程系列原理篇--JDK中的通信工具类Semaphore

    前言 java多线程之间进行通信时,JDK主要提供了以下几种通信工具类.主要有Semaphore.CountDownLatch.CyclicBarrier.exchanger.Phaser这几个通讯类 ...

  4. 学习ASP.NET Core Blazor编程系列十——路由(中)

    学习ASP.NET Core Blazor编程系列一--综述 学习ASP.NET Core Blazor编程系列二--第一个Blazor应用程序(上) 学习ASP.NET Core Blazor编程系 ...

  5. 游戏编程系列[1]--游戏编程中RPC协议的使用

    RPC(Remote Procedure Call Protocol)--远程过程调用协议,它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议.RPC协议假定某些传输协议的存在 ...

  6. 游戏编程系列[1]--游戏编程中RPC协议的使用[3]--体验

    运行环境,客户端一般编译为.Net 3.5 Unity兼容,服务端因为用了一些库,所以一般为4.0 或往上.同一份代码,建立拥有2个项目.客户端引用: WindNet.Client服务端引用: OpL ...

  7. 游戏编程系列[1]--游戏编程中RPC协议的使用[2]--Aop PostSharp篇

    上一篇我们使用了一个通用JSON协议约定来进行达到远程调用的目的.但是从实现上,我们需要不断的在所有的方法上添加拦截,并且判断拦截,然后执行,这就达到了一个比较繁琐的目的. 之前我们尝试过使用代码生成 ...

  8. python编程系列---Pycharm快捷键(更新中....)

    以下是我常用到的Pycharm快捷键(还有很多,只是我暂时用的最多的就这些): 在开发过程中,经常使用一些快捷键会大大提高开发效率,不要因为看这多而不用,常用的就那些,用得多就都记住了,脱离鼠标,逼格 ...

  9. netty系列之:在netty中使用native传输协议

    目录 简介 native传输协议的依赖 netty本地传输协议的使用 总结 简介 对于IO来说,除了传统的block IO,使用最多的就是NIO了,通常我们在netty程序中最常用到的就是NIO,比如 ...

随机推荐

  1. shell运算符

    原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用. expr 是一款表达式计算工具,使用它能完成表达式的求值操作. #!/bin/bash v ...

  2. ASP.NET Core 中文文档 第四章 MVC(3.8)视图中的依赖注入

    原文:Dependency injection into views 作者:Steve Smith 翻译:姚阿勇(Dr.Yao) 校对:孟帅洋(书缘) ASP.NET Core 支持在视图中使用 依赖 ...

  3. 6. ModelDriven拦截器、Preparable 拦截器

    1. 问题 Struts2 的 Action 我们将它定义为一个控制器,但是由于在 Action 中也可以来编写一些业务逻辑,也有人会在 Action 输入业务逻辑层. 但是在企业开发中,我们一般会将 ...

  4. 如何用Java类配置Spring MVC(不通过web.xml和XML方式)

    DispatcherServlet是Spring MVC的核心,按照传统方式, 需要把它配置到web.xml中. 我个人比较不喜欢XML配置方式, XML看起来太累, 冗长繁琐. 还好借助于Servl ...

  5. html中返回上一页的各种写法【转】

    超链接返回上一页代码: <a href="#" onClick="javascript :history.back(-1);">返回上一页</ ...

  6. Angular2 Hello World 之 RC6

    angular2还没有发布正式版,确实有点不靠谱,变化太频繁,之前写的demo直接将js升级到最新版之后发现就不能用了……所以现在在写一篇demo——基于RC6.参考:http://web3.code ...

  7. maven 快照

    大型应用软件一般由多个模块组成,一般它是多个团队开发同一个应用程序的不同模块,这是比较常见的场景.例如,一个团队正在对应用程序的应用程序,用户界面项目(app-ui.jar:1.0) 的前端进行开发, ...

  8. Linux虚拟化学习笔记<一>

    关于虚拟化,原理的东西是非常复杂的,要想完全理解,没有足够的耐心是不不能完全学透这部分内容的.那下面我主要以资源汇总的形式把一些资料罗列出来,帮助大家快速理解虚拟化,快速使用和配置. 为什么要虚拟化: ...

  9. 洛谷P1547 Out of Hay

    题目背景 奶牛爱干草 题目描述 Bessie 计划调查N (2 <= N <= 2,000)个农场的干草情况,它从1号农场出发.农场之间总共有M (1 <= M <= 10,0 ...

  10. 纯JS打造比QQ空间更强大的图片浏览器-支持拖拽、缩放、过滤、缩略图等

    在线演示地址(打开网页后,点击商家图册): http://www.sport7.cn/cc/jiangnan/football5.html 先看一看效果图: 该图片浏览器实现的功能如下: 1. 鼠标滚 ...