根据设计生成两个接口,IRpcSend send方法返回数据要求包装成QResult对象

public interface IRpcSend {
public <T> QResult<T> send(byte command, Object... args);
}
public interface IRpcReceive {
public <T> T receive(byte command, Objec[] args);
}
public interface IRpcContext {
RpcContext getContext();
void setContext(RpcContext ctx);
}
public interface QResult<T> extends Future<T> {

    /**
* 是否出错
* */
public boolean isError();
/**
* 获取返回结果
* */
public T getResult();
/**
* 设置结果
* */
public void setResult(T result);
}
 @SuppressWarnings("unchecked")
public abstract class QRpcFactory {
private static Method SEND_METHOD = ReflectUtil.getMethod(IRpcSend.class, "send");
private static Method RECEIVE_METHOD = ReflectUtil.getMethod(IRpcReceive.class, "receive"); private static Map<Short, Class<?>> SEND_CLASS = new HashMap<>();
private static Map<Short, IRpcReceive> RECEIVE = new HashMap<>(); public static <T> T loadSendProxy(Class<T> target, QNode... nodes) {
T ret = loadSendPorxy0(target);
IRpcContext ctx = (IRpcContext) ret;
ctx.setContext(RpcContext.of(nodes));
return ret;
} public static <T> T loadSendProxy(Class<T> target, Long... ids) {
T ret = loadSendPorxy0(target);
IRpcContext ctx = (IRpcContext) ret;
ctx.setContext(RpcContext.of(ids));
return ret;
} public static <T> T loadSendProxy(Class<T> target, String... addresses) {
T ret = loadSendPorxy0(target);
IRpcContext ctx = (IRpcContext) ret;
ctx.setContext(RpcContext.of(addresses));
return ret;
} private static <T> T loadSendPorxy0(Class<T> target) {
QModel modelAnno = ReflectUtil.getAnno(target, QModel.class);
Class<?> proxyClass = SEND_CLASS.get(modelAnno.value());
T ret = null;
try {
ret = (T) proxyClass.newInstance();
} catch (InstantiationException | IllegalAccessException e) {
throw new RuntimeException(e);
}
return ret;
} public static <T> T loadLocalProxy(Class<T> target) {
QModel modelAnno = ReflectUtil.getAnno(target, QModel.class);
Object ret = RECEIVE.get(modelAnno.value());
return (T) ret;
} public static IRpcReceive loadReceiveProxy(short model) {
IRpcReceive ret = RECEIVE.get(model);
return ret;
}
}

QRpcFactory

 public static void registerSendProxy(Class<?> target) {
if (!target.isInterface()) {
throw new RuntimeException("class is not Interface : " + target);
}
QModel modelAnno = ReflectUtil.getAnno(target, QModel.class);
String proxyClassName = target.getCanonicalName() + "$$send$$";
ClassPool classPool = JavassistHepler.classPool;
CtClass ctClass = classPool.makeClass(proxyClassName); try {
// 设置接口
CtClass[] interfaces = new CtClass[3];
interfaces[0] = classPool.get(target.getName());
interfaces[1] = classPool.get(IRpcSend.class.getName());
interfaces[2] = classPool.get(IRpcContext.class.getName());
ctClass.setInterfaces(interfaces); {
// 添加ctx字段
final String ctxName = RpcContext.class.getName();
CtField ctField = new CtField(classPool.get(ctxName), "ctx", ctClass);
ctField.setModifiers(Modifier.PRIVATE);
ctClass.addField(ctField);
// 添加ctx get set 方法
CtMethod ctMethod = CtMethod.make("public " + ctxName + " getContext(){return ctx;}", ctClass);
ctMethod.setModifiers(Modifier.PUBLIC);
ctClass.addMethod(ctMethod); ctMethod = CtMethod.make("public void setContext(" + ctxName + " value){ ctx =value;}", ctClass);
ctMethod.setModifiers(Modifier.PUBLIC);
ctClass.addMethod(ctMethod);
} // 生成send method 调用静态方法减少书写复杂
{
final String body = " return ($r){ " + resultType + QRpcFactory.class.getName() + ".proxySend(this,$2, (short)" + modelAnno.value() + " ,(byte) $1);}";
JavassistHepler.addMethod(ctClass, SEND_METHOD, body);
} // 生成代理方法
for (Method method : target.getDeclaredMethods()) {
QCommond commond = method.getAnnotation(QCommond.class);
if (commond == null) {
continue;
} String resultType = "";
if (void.class != method.getReturnType()) {
resultType = " return ($r) ";
}
final String body = "{ " + resultType + " this.send((byte)" + commond.value() + ",$args); }";
JavassistHepler.addMethod(ctClass, method, body);
} // 保存记录
Class<?> newClass = ctClass.toClass();
ctClass.detach();
SEND_CLASS.put(modelAnno.value(), newClass); } catch (Exception e) {
e.printStackTrace();
}
}
public static QResult proxySend(IRpcContext target, Object[] args, short model, byte commondIndex) {
System.out.println("model : " + model + " commondIndex : " + commondIndex);
return null;
}

测试理论:

    @QModel(1)
public interface TestObject {
@QCommond(1)
public void a(int a, String b); @QCommond(2)
public void setAge(int value); @QCommond(3)
public QResult<Integer> getAge(); @QCommond(4)
public void setObj(TestObject1 obj); @QCommond(5)
public void test(int a,Integer b,Double c ,List<Integer> d,Integer[] e );
} public class TestObject1 {
public int a = 67;
} public static class TestObjectImpl implements TestObject {
private int age; @Override
public void a(int a, String b) {
System.out.println("a : " + a + " " + b);
} @Override
public void setAge(int value) {
age = value;
} @Override
public QResult<Integer> getAge() {
return QResultWrapper.of(age);
} @Override
public void setObj(TestObject1 obj) {
System.out.println(obj.a);
} @Override
public void test(int a, Integer b, Double c ,List<Integer> d,Integer[] e ) { }
}
 public class TestRpcProxy {
@Test
public void testSend() {
QRpcFactory3.registerSendProxy(TestObject.class);
TestObject proxy = QRpcFactory3.loadSendProxy(TestObject.class, 1L); proxy.a(1, "b");
QResult<Integer> ret = proxy.getAge();
}
}

由于发送处理涉及netty,先不考虑通信处理实现,在实现写框架过程,依赖越少实现越简单。所以每个实例对象构造出来时必须能正常工作

[编织消息框架][JAVA核心技术]动态代理应用7-IRpcSend实现的更多相关文章

  1. [编织消息框架][JAVA核心技术]动态代理介绍

    由于java是种强类型静态语言,在执行时无法动态生成代码,静态语言基本都有这特性 动态生成代码有几种好处,也是弱类型语言的优点 1.部份逻辑可以实现热更新 2.远程调用实现非常适合 3.能动态生成扩展 ...

  2. [编织消息框架][JAVA核心技术]动态代理应用12-总结

    动态代理这篇比较长,是框架组成的重要基础 回顾下学到的应用技术 1.异常应用 2.annotation技术 3.数值与逻辑分享 4.jdk.cglib.javassist等动态代理技术 5.懒处理.预 ...

  3. [编织消息框架][JAVA核心技术]动态代理应用4

    基础部份: 接下来讲编译JAVA时,生成自定义class 我们用 javax.annotation.processing.AbstractProcessor 来处理 public abstract c ...

  4. [编织消息框架][JAVA核心技术]动态代理应用8-IRpcReceive实现

    private static Map<Short, Map<Byte, Method>> RECEIVE_METHOD_INFO = new HashMap<>() ...

  5. [编织消息框架][JAVA核心技术]动态代理应用4-annotationProcessor

    基础部份: 接下来讲编译JAVA时,生成自定义class 我们用 javax.annotation.processing.AbstractProcessor 来处理 public abstract c ...

  6. [编织消息框架][JAVA核心技术]动态代理应用2

    接下来如何实现 第一步:先把服务类,调用方法转换成数字,方便传输 第二步:提取元信息,提取又有三种方式,三种各有优点,最优方式是第一种 1.编译java时处理 2.程序启动时处理,预处理 3.调用时处 ...

  7. [编织消息框架][JAVA核心技术]动态代理应用5-javassist

    基础部份: 修改class我们用到javassist,在pom.xml添加 <properties> <javassist.version>3.18.2-GA</java ...

  8. [编织消息框架][JAVA核心技术]动态代理应用9-扫描class

    之前介绍的annotationProcessor能在编译时生成自定义class,但有个缺点,只能每次添加/删除java文件才会执行,那天换了个人不清楚就坑大了 还记得之前介绍的编译时处理,懒处理,还有 ...

  9. [编织消息框架][JAVA核心技术]动态代理应用10-水平扩展方案

    服务分为系统服务同用户服务两种 水平扩展是基于系统服务,而拆分方式又有几种方案,按数据跟业务情况来做决策 1.每个服务独立存储(图1):每个服务只负责一个或多个领域实体存储,A服务不能直接修改B服务的 ...

随机推荐

  1. Scala入门系列(一):基础语法

    Scala基础语法 Scala与JAVA的关系 Scala是基于Java虚拟机,也就是JVM的一门编程语言,所有Scala的代码都需要经过编译为字节码,然后交由Java虚拟机来运行. 所以Scala和 ...

  2. poj 2769 Reduced ID Numbers 同余定理

    链接:http://poj.org/problem?id=2769 题意:寻找数m,是的对于n个数的余数不同 思路:暴力,优化:同余部分不用测试 代码: #include <iostream&g ...

  3. angularjs 给封装的模态框元素传值,和实现兄弟传值

    本例实现封装的元素所放的位置不同,而选择不同的传值,这里举例封装了bootstrap模态框,以后也方便大家去直接使用.方法举例如下:首先主页调用css/js有: <link rel=" ...

  4. 简陋的斗地主,js实现

    最近闲了两天没事做,用js写了个斗地主,练习练习.代码和功能都很简陋,还有bug,咋只是聊聊自己的思路. 这里说说斗地主主要包含的功能:洗牌,发牌,玩家出牌.电脑出牌,出牌规则的验证,输赢啥的没有判断 ...

  5. 前端备忘录--JQuery选择器

    基本选择器 基本选择器是最常用的选择器,也是最简单的选择器. $("#test") //选取id为test的元素 $(".test") //选取class为te ...

  6. 深入浅出了解frame和bounds

    frame frame的官方解释如下: The frame rectangle, which describes the view's location and size in its supervi ...

  7. Android学习笔记(10).布局管理器

    布局管理器的几个类都是ViewGroup派生的,用于管理组件的分布和大小,使用布局管理器能够非常好地解决屏幕适配问题. 布局管理器本身也是一个UI组件,布局管理器能够相互嵌套使用,以下是布局管理器的类 ...

  8. 给新手--安装tomcat后username和password设置以及项目怎么部署在tomcatserver上

    安装后tomcatserver后.登陆首先就是让输入username和password.但是我们在安装tomcat的过程中好像没有让设置username和password,这时候可能有人就抓狂了.还有 ...

  9. ASP.MVC当URL跳转时候参数的安全性

    一个页面跳转到另外一个页面直接将参数写在URL上面并不安全比如 http://XXXXXXXXXXX/meeting/shakeGroup?id=5381&uid=o0En_sj1J0bFgI ...

  10. Python的下划线_

    1.单下划线(_) 通常情况下,单下划线(_)会在以下3种场景中使用: 1.1 在解释器中: 在这种情况下,"_"代表交互式解释器会话中上一条执行的语句的结果.这种用法首先被标准C ...