dubbo功能很是完善,不少时候咱们不须要重复造轮子,下面列举一些你不必定知道,可是很好用的功能;java
在开发及测试环境下,可能须要绕过注册中心,只测试指定服务提供者,这时候可能须要点对点直连,点对点直连模式,将以服务接口为单位,忽略注册中心的提供者列表,A 接口配置点对点,不影响 B 接口从注册中心获取列表(说明:官方只建议开发&测试环境使用该功能),用法以下,url指定的地址就是直连地址:spring
<dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" version="1.0.0" url="dubbo://172.18.1.205:20888/" />复制代码
当一个接口实现,出现不兼容升级时,能够用版本号过渡,版本号不一样的服务相互间不引用,用法以下:缓存
<dubbo:service interface="com.alibaba.dubbo.demo.DemoService" ref="demoService" version="1.0.0" />复制代码
利用dubbo该特性,咱们可以实现一些功能的灰度发布,实现步骤以下:bash
这样定义Provider和Consumer后,新旧接口实现各承担50%
的流量;app
利用dubbo该特性,还能完成不兼容版本迁移:框架
回声测试用于检测服务是否可用,回声测试按照正常请求流程执行,可以测试整个调用是否通畅,可用于监控。
全部服务自动实现EchoService接口,只需将任意服务引用强制转型为EchoService 便可使用,使用方式(demoService是spring管理的bean)ide
EchoService echoService = (EchoService) demoService;
System.out.println(echoService.$echo("hello"));复制代码
能够经过 RpcContext 的 setAttachment()
和 getAttachment()
在Consumer和Provider之间进行参数的隐式传递,例如Controller层拦截登陆token,把根据token获得的memberId传给dubbo服务就能使用隐式参数传递的方式,setAttachment()
设置的 KV 对,在完成一次远程调用会被清空,即屡次远程调用要屡次设置。使用方式:测试
RpcContext.getContext().setAttachment("CRT_MEMBER_ID", "13828886888");复制代码
RpcContext.getContext().getAttachment("CRT_MEMBER_ID")复制代码
上下文中存放的是当前调用过程当中所需的环境信息。全部配置信息都将转换为 URL 的参数
RpcContext 是一个 ThreadLocal 的临时状态记录器,当接收到 RPC 请求,或发起 RPC 请求时,RpcContext 的状态
都会变化。例如:A 调 B,B 再调 C,则 B 机器上,在 B 调 C 以前,RpcContext 记录的是 A 调 B 的信息,在 B 调 C
以后,RpcContext 记录的是 B 调 C 的信息。使用方式:ui
boolean isConsumerSide = RpcContext.getContext().isConsumerSide();复制代码
本地假装一般用于服务降级,例如某验权服务,当服务提供方所有挂掉后,客户端不抛出异常,而是经过 Mock 数据
返回受权失败。使用方式以下,mock指定的实现类在Provider抛出RpcException异常时执行(必定要抛出RpcException异常才执行),取代远程返回结果:url
<dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" version="1.0.0" mock="com.alibaba.dubbo.demo.consumer.mock.DemoServiceMock"/>复制代码
DemoServiceMock实现源码:
public class DemoServiceMock implements DemoService {
public String sayHello(String name) {
return "mock-value";
}
}复制代码
泛化接口调用方式主要用于客户端没有 API 接口及模型类元的状况,参数及返回值中的全部 POJO 均用Map表示,一般用于框架集成,例如:实现一个通用的服务测试框架,可经过GenericService调用全部服务实现。使用方式:
<dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" generic="true"/>复制代码
调用源码:
/**
* @author afei
* @version 1.0.0
* @since 2017年11月22日
*/
public class Main {
public static void main(String[] args) {
// 引⽤远程服务, 该实例⾥⾯封装了全部与注册中⼼及服务提供⽅链接,请缓存
ReferenceConfig<GenericService> reference = new ReferenceConfig<GenericService>();
// 弱类型接⼝名
reference.setInterface("com.alibaba.dubbo.demo.DemoService");
reference.setVersion("1.0.0");
// 声明为泛化接⼝
reference.setGeneric(true);
// ⽤com.alibaba.dubbo.rpc.service.GenericService能够替代全部接口引用⽤
GenericService genericService = reference.get();
// 基本类型以及Date,List,Map等不须要转换,直接调⽤
Object result = genericService.$invoke("sayYes", new String[] {
"java.lang.String"
},
new Object[] {"afei"});
System.out.println("result --> "+result);
// ⽤Map表示POJO参数,若是返回值为POJO也将自动转成Map
Map<String, Object> teacher = new HashMap<String, Object>();
teacher.put("id", "1");
teacher.put("name", "admin");
teacher.put("age", "18");
teacher.put("level", "3");
teacher.put("remark", "测试");
// 若是返回POJO将自动转成Map
result = genericService.$invoke("justTest", new String[]
{"com.alibaba.dubbo.demo.bean.HighTeacher"}, new Object[]{teacher}); System.out.println("result --> "+result);
}
}复制代码
若是想记录每次请求信息,可开启访问日志,相似于Ngnix的访问日志。注意:此日志量比较大,请注意磁盘容量。使用方式(若是配置局部,全局访问日志就会失效):
配置全局:
<dubbo:provider accesslog="/app/dubbo-demo.log"/>复制代码
配置局部:
<dubbo:service interface="com.alibaba.dubbo.demo.DemoService" ref="demoService" accesslog="/app/demo.log"/> <dubbo:service interface="com.alibaba.dubbo.demo.TestService" ref="testService" accesslog="/app/test.log"/>复制代码
日志格式样式:
[2017-11-22 10:23:20] 172.18.1.205:56144 -> 172.18.1.205:20886 - com.alibaba.dubbo.demo.DemoService:1.0.0 sayHello(java.lang.String) ["afei"]复制代码
若是服务须要预热时间,好比初始化本地缓存,等待相关资源就位等,可使用delay进行延迟暴露。使Dubbo在Spring容器初始化完后延迟多少毫秒再暴露服务。使用方式:
<dubbo:provider delay="5000"/>复制代码
或者:
<dubbo:service delay="5000" interface="com.alibaba.dubbo.demo.DemoService" ref="demoService" version="1.0.0"/>复制代码