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

之前我们尝试过使用代码生成器,直接生成。

构造基类,TestClassA
然后使用代码生成器,构造第二个类,TestClassAProxy。
所有需要的方法,我们都采用虚函数,然后重载,直接调用Base.Func();
这样也就完成了任务,这个方法的好处,代码生成简单,但是需要替换使用类别。
相对来说实现成本还比较低,对下面说的方法没什么信心的同学,可以采用这种实现方式。

然后我们说到到了本篇的主角,AOP利器 PostSharp
(额,商业软件,但是45天免费试用,我不会告诉你删了注册表后可以无限免费试用)

来看例子:

   [Serializable]
//[DebuggerNonUserCode]
public sealed class ProxyAttribute : OnMethodBoundaryAspect
{ //[DebuggerNonUserCode]
public override void OnException(MethodExecutionArgs args)
{
base.OnException(args);
WindNet.RemoteCall.Instance.DoError(args.Exception);
args.FlowBehavior = FlowBehavior.Continue;
//不抛出 } //[DebuggerNonUserCode]
public override void OnSuccess(MethodExecutionArgs args)
{
base.OnSuccess(args);
WindNet.RemoteCall.Instance.DoCallBack(args.ReturnValue, __cid);
} private WindNet.RPC.RequestObj __cid; //[DebuggerNonUserCode]
public override void OnEntry(MethodExecutionArgs args)
{
var callback = args.Arguments.Last(); var methodName =args.Method.DeclaringType.Name + "." + args.Method.Name;
var outargs = new List<object>();
for (int i = 0; i < args.Arguments.Count - 1; i++)
{
outargs.Add(args.Arguments[i]);
}
__cid = WindNet.RemoteCall.Instance.DoCall(callback as Delegate, args.Instance as IOpItem, methodName, outargs.ToArray()); if (WindNet.RemoteCall.Instance.IsLocal || __cid.isLocal)
{
//继续执行
}
else
{
args.FlowBehavior = FlowBehavior.Return;
} }
}

  然后我们在之前的方法上添加:

	  public class LocalTest
{
/// <summary>
/// 添加一个消息
/// </summary>
[Proxy]
public static void AddMessage(string topic, string messageBody)
{
//这里什么都不干
}
}

  生成后反编译

public class LocalTest
{
public static void AddMessage(string topic, string messageBody)
{
MethodExecutionArgs methodExecutionArgs = new MethodExecutionArgs(null, new Arguments<string, string>
{
Arg0 = topic,
Arg1 = messageBody
});
MethodExecutionArgs arg_25_0 = methodExecutionArgs;
MethodBase = <>z__a_3._2;
arg_25_0.Method = ;
<>z__a_3.a2.OnEntry(methodExecutionArgs);
if (methodExecutionArgs.FlowBehavior != FlowBehavior.Return)
{
try
{
<>z__a_3.a2.OnSuccess(methodExecutionArgs);
}
catch (Exception exception)
{
methodExecutionArgs.Exception = exception;
<>z__a_3.a2.OnException(methodExecutionArgs);
switch (methodExecutionArgs.FlowBehavior)
{
case FlowBehavior.Default:
case FlowBehavior.RethrowException:
IL_81:
throw;
case FlowBehavior.Continue:
methodExecutionArgs.Exception = null;
return;
case FlowBehavior.Return:
methodExecutionArgs.Exception = null;
return;
case FlowBehavior.ThrowException:
throw methodExecutionArgs.Exception;
}
goto IL_81;
}
}
}
}

  

帮我们正确的填写了缺失的部分。

优点:
简便,自动化,并且不需要去修改成代理类。
商业版,稳定性良好。
在VS中集成

缺点:
商业化,如果上线保险点还是去买一个。
Unity中,调试有点Bug,Pdb貌似有bug,兼容性问题。
如果不想用DLL,需要一些特殊手法才能在Unity中使用。

PS:不喜欢用商业产品,推荐用Fody,也有实现,回头有可能会切换到那个版本去。

游戏编程系列[1]--游戏编程中RPC协议的使用[2]--Aop PostSharp篇的更多相关文章

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

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

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

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

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

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

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

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

  5. 游戏编程系列[2]--游戏编程中RPC与OpLog协议的结合--序

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

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

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

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

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

  8. Hadoop中RPC协议小例子报错java.lang.reflect.UndeclaredThrowableException解决方法

    最近在学习传智播客吴超老师的Hadoop视频,里面他在讲解RPC通信原理的过程中给了一个RPC的小例子,但是自己编写的过程中遇到一个小错误,整理如下: log4j:WARN No appenders ...

  9. Java并发编程系列-(9) JDK 8/9/10中的并发

    9.1 CompletableFuture CompletableFuture是JDK 8中引入的工具类,实现了Future接口,对以往的FutureTask的功能进行了增强. 手动设置完成状态 Co ...

随机推荐

  1. 使用Visual Studio 2015 开发ASP.NET MVC 5 项目部署到Mono/Jexus

    最新的Mono 4.4已经支持运行asp.net mvc5项目,有的同学听了这句话就兴高采烈的拿起Visual Studio 2015创建了一个mvc 5的项目,然后部署到Mono上,浏览下发现一堆错 ...

  2. Java 征途:行者的地图

    前段时间应因缘梳理了下自己的 Java 知识体系, 成文一篇望能帮到即将走进或正在 Java 世界跋涉的程序员们. 第一张,基础图 大约在 2003 年我开始知道 Java 的(当时还在用 Delph ...

  3. ABP教程-打造一个《电话簿项目》-目录-MPA版本-基于ABP1.13版本

    此系列文章会进行不定期的更新,应该会有6章左右. 感兴趣的朋友可以跟着看看,本教程适合已经看过ABP的文档但是又无从下手的小伙伴们. 初衷: 发布系列教程的原因是发现ABP在园子火了很久,但是发现还是 ...

  4. TODO:Laravel 使用blade标签布局页面

    TODO:Laravel 使用blade标签布局页面 本文主要介绍Laravel的标签使用,统一布局页面.主要用到到标签有@yield,@ stack,@extends,@section,@stop, ...

  5. 水印第三版 ~ 变态水印(这次用Magick.NET来实现,附需求分析和源码)

    技能 汇总:http://www.cnblogs.com/dunitian/p/4822808.html#skill 以前的水印,只是简单走起,用的是原生态的方法.现在各种变态水印,于是就不再用原生态 ...

  6. Android业务组件化之URL Scheme使用

    前言: 最近公司业务发展迅速,单一的项目工程不再适合公司发展需要,所以开始推进公司APP业务组件化,很荣幸自己能够牵头做这件事,经过研究实现组件化的通信方案通过URL Scheme,所以想着现在还是在 ...

  7. 代码的坏味道(19)——狎昵关系(Inappropriate Intimacy)

    坏味道--狎昵关系(Inappropriate Intimacy) 特征 一个类大量使用另一个类的内部字段和方法. 问题原因 类和类之间应该尽量少的感知彼此(减少耦合).这样的类更容易维护和复用. 解 ...

  8. VS2015常用快捷键总结

    生成解决方案 F6,生成项目Shift+F6 调试执行F5,终止调试执行Shift+F5 执行调试Ctrl+F5 查找下一个F3,查找上一个Shift+F3 附加到进程Ctrl+Alt+P,逐过程F1 ...

  9. web服务器集群

    概述 集群和分布式都是从集中式进化而来的.分布式和集群会相互合作的,同时的集群和分布式.在这里重点说说集群 集群是什么? 集群能提高单位时间内处理的任务数量,提升服务器性能 有多台服务器去处理任务,但 ...

  10. webpack学习总结

    前言 在还未接触webpack,就有几个疑问: 1. webpack本质上是什么? 2. 跟异步模块加载有关系吗? 3. 可否生成多个文件,一定是一个? 4. 被引用的文件有其他异步加载模块怎么办? ...