[编织消息框架][JAVA核心技术]动态代理应用7-IRpcSend实现
根据设计生成两个接口,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实现的更多相关文章
- [编织消息框架][JAVA核心技术]动态代理介绍
由于java是种强类型静态语言,在执行时无法动态生成代码,静态语言基本都有这特性 动态生成代码有几种好处,也是弱类型语言的优点 1.部份逻辑可以实现热更新 2.远程调用实现非常适合 3.能动态生成扩展 ...
- [编织消息框架][JAVA核心技术]动态代理应用12-总结
动态代理这篇比较长,是框架组成的重要基础 回顾下学到的应用技术 1.异常应用 2.annotation技术 3.数值与逻辑分享 4.jdk.cglib.javassist等动态代理技术 5.懒处理.预 ...
- [编织消息框架][JAVA核心技术]动态代理应用4
基础部份: 接下来讲编译JAVA时,生成自定义class 我们用 javax.annotation.processing.AbstractProcessor 来处理 public abstract c ...
- [编织消息框架][JAVA核心技术]动态代理应用8-IRpcReceive实现
private static Map<Short, Map<Byte, Method>> RECEIVE_METHOD_INFO = new HashMap<>() ...
- [编织消息框架][JAVA核心技术]动态代理应用4-annotationProcessor
基础部份: 接下来讲编译JAVA时,生成自定义class 我们用 javax.annotation.processing.AbstractProcessor 来处理 public abstract c ...
- [编织消息框架][JAVA核心技术]动态代理应用2
接下来如何实现 第一步:先把服务类,调用方法转换成数字,方便传输 第二步:提取元信息,提取又有三种方式,三种各有优点,最优方式是第一种 1.编译java时处理 2.程序启动时处理,预处理 3.调用时处 ...
- [编织消息框架][JAVA核心技术]动态代理应用5-javassist
基础部份: 修改class我们用到javassist,在pom.xml添加 <properties> <javassist.version>3.18.2-GA</java ...
- [编织消息框架][JAVA核心技术]动态代理应用9-扫描class
之前介绍的annotationProcessor能在编译时生成自定义class,但有个缺点,只能每次添加/删除java文件才会执行,那天换了个人不清楚就坑大了 还记得之前介绍的编译时处理,懒处理,还有 ...
- [编织消息框架][JAVA核心技术]动态代理应用10-水平扩展方案
服务分为系统服务同用户服务两种 水平扩展是基于系统服务,而拆分方式又有几种方案,按数据跟业务情况来做决策 1.每个服务独立存储(图1):每个服务只负责一个或多个领域实体存储,A服务不能直接修改B服务的 ...
随机推荐
- thinkphp5源码解析(1)数据库
前言 tp5的数据库操作全部通过Db类完成,比较符合国人的习惯,比如简单的Db::query().Db::execute(),还有复杂的链式操作Db::where('id=1')->select ...
- CCF-201312-3-最大的矩形
问题描述 试题编号: 201312-3 试题名称: 最大的矩形 时间限制: 1.0s 内存限制: 256.0MB 问题描述: 问题描述 在横轴上放了n个相邻的矩形,每个矩形的宽度是1,而第i(1 ≤ ...
- php命令执行脚本
php -f jiaoben.php & 读入并解释指明的文件.
- php面试之数据结构和算法
二.数据结构和算法 1.使对象可以像数组一样进行foreach循环,要求属性必须是私有.(Iterator模式的PHP5实现,写一类实现Iterator接口)(腾讯) <?php class T ...
- 修复mysql表
1>用"repair table"方式修复语法:repair table 表名 [选项]选项如下:QUICK 用在数据表还没被修改的情况下,速度最快EXTENDED 试图去恢 ...
- day1-Python入门
百度云有关文档资料链接 链接:https://pan.baidu.com/s/1pLighnX 密码:j69s
- 巧用第三方高速开发Android App 热门第三方SDK及框架
巧用第三方高速开发Android App 热门第三方SDK及框架 历经大半年的时间,最终是把这门课程给录制出来了,也就在今天,正式在慕课网上上线了 项目地址:巧用第三方高速开发Android App ...
- sqlserver2005公布与订阅配置步骤
1,新建公布 前提条件:第一要调通网络,在sqlserver configuration manager 中选择mssqlserver的协议把named pipes改为启用.第二要建立一个目录D:\b ...
- Hibernate学习(二补充)关系映射----基于外键的双向一对一
刚刚写的是基于外键的单向一对一. 那么双向一对一就是在单向一对一的基础上稍微改动就可以了. account.java和account.hbm.xml都不用变动 只要我们小小的变动address.j ...
- C#开发微信门户及应用(48) - 在微信框架中整合CacheManager 缓存框架
在我们的很多框架或者项目应用中,缓存在一定程度上可以提高程序的响应速度,以及减轻服务器的承载压力,因此在一些地方我们都考虑引入缓存模块,这篇随笔介绍使用开源缓存框架CacheManager来实现数据的 ...