Check启动检查

根据之前的学习,我们简单理解的Dubbo远程调用的基本流程,服务提供者注册到注册中心,然后服务消费者通过监听注册中心达到远程调用的目的,那么如果注册中心中没有消费者对应的接口会怎么样呢?

开启Zookeeper,在不运行服务提供者的情况下直接运行消费者,结果是:

// 抛出了异常信息
java.lang.reflect.InvocationTargetException: null
// 其中可以看到有价值的异常信息为:
Failed to check the status of the service site.hanzhe.service.UserService. No provider available for the service site.hanzhe.service.UserService from the url zookeeper://127.0.0.1:2181/com.alibaba.dubbo ......

大致含义为我们通过@Reference注解进行远程调用,但是注册中心中并没有对应的服务所以报错

这样依赖我们就知道了远程调用是在程序初始化的时候就已经与远程服务建立连接了,现在我们不希望它初始化时连接,我们希望在真正用到它的时候再进行连接,这样我们提供者就可以随时注册上去提供服务,想要实现这个功能可以利用注解中的check属性:

@RestController
public class UserController {
@Reference(check = false)
private UserService userService; @GetMapping("/selectUser")
public List<UserEntity> selectUse() {
return userService.selectList();
}
}

check属性为启动检查,默认值为true,当我们设置为false之后启动就不报错了,然后随时启动提供者进行远程调用

通过设置@Referencecheck属性可以关闭启动检查,但是如果我们程序中存在着大量的远程接口就不方便操作了,这里Dubbo为我们提供了一个全局的消费者配置:

# 通过设置dubbo.consumer.check属性来设置消费者所有远程接口的启动检查
dubbo:
consumer:
check: false

这时候我们在移除@Referencecheck属性,也可以实现相同的功能,且一劳永逸!

Timeout超时时间

Dubbo是一款RPC远程调用框架,远程调用肯定离不开网络,那么如果在远程调用的时候时间过长应该怎么处理?

做一个测试,在服务提供者中返回结果之前做一个阻塞效果:

@Service
@com.alibaba.dubbo.config.annotation.Service
public class UserServiceImpl implements UserService { @Override
public List<UserEntity> selectList() {
UserEntity entity1 = new UserEntity("马保国", "69", "男");
UserEntity entity2 = new UserEntity("张全蛋", "31", "男");
try {
// 这里让该线程睡2秒钟
Thread.sleep(2000);
} catch (InterruptedException e) { }
return Arrays.asList(entity1, entity2);
} }

然后在消费者进行远程调用测试,抛出了如下异常:

com.alibaba.dubbo.remoting.TimeoutException: Waiting server-side response timeout by scan timer ...

简单来说就是服务端响应太慢请求超时了,通过查看官网可知道默认的超时时间为1000ms,也就是说当请求超过1秒就会被抛弃,但是有些时候1秒有些太短,这里在消费者方通过@Referencetimeout属性手动设置超时时间:

@RestController
public class UserController {
@Reference(check = false, timeout = 3000)
private UserService userService; @GetMapping("/selectUser")
public List<UserEntity> selectUse() {
return userService.selectList();
}
}

在远程调用时设置超时时间为三秒,这样一来就可以调用成功了!和检查属性一样,超时时间也可以全局设置:

# 通过设置dubbo.consumer.check属性来设置消费者所有远程接口的启动检查
dubbo:
consumer:
check: false
timeout: 3000

Retries请求重发

我们可以通过延长请求超时时间来提高稳定率,但如果出现了意外情况在规定的时间内也没能完成操作,那么我们就需要进行重发进行操作了:

@RestController
public class UserController {
// retries当请求失败后,重新发送该请求,设置重发次数最多为2次
// 加上本身的一次也就是一共3次请求,如果3次都失败那就是真正的失败
@Reference(check = false, retries=2)
private UserService userService; @GetMapping("/selectUser")
public List<UserEntity> selectUse() {
return userService.selectList();
}
}

我们在服务提供者中使用打印语句来观察请求的次数,可以看到的确调用了三次:

关于请求重发

并不是所有请求都可以进行请求重发的,原则上只有幂等可以设置重发,非幂等不可以设置重发

  • 幂等:类似delete、update、select操作多次执行并不会对数据造成影响
  • 非幂等:类似inert操作多次执行可能会造成不可控的后果

当然这只是原则,实际上还需要根据业务逻辑来判断是否可以使用,例如UPDATE user SET number = number + #{number} .. 就不适合重发

Version多版本

服务提供者通过实现接口,暴露出接口的实现类,从而在消费者端实现远程调用,在服务者暴露接口时可以针对某一个接口暴露多个实现类,每个实现类有着自己的版本号,通过版本号来控制调用哪个实现

@Service
// 设置当前service实现版本为v1
@com.alibaba.dubbo.config.annotation.Service(version="v1")
public class UserServiceImpl implements UserService { @Override
public List<UserEntity> selectList() {
UserEntity entity1 = new UserEntity("马保国", "69", "男");
UserEntity entity2 = new UserEntity("张全蛋", "31", "男");
return Arrays.asList(entity1, entity2);
} } @Service
// 复制一个实现,修改版本为v2,并修改数据内容
@com.alibaba.dubbo.config.annotation.Service(version="v1")
public class UserServiceImpl implements UserService { @Override
public List<UserEntity> selectList() {
UserEntity entity1 = new UserEntity("马保国", "69", "女");
UserEntity entity2 = new UserEntity("张全蛋", "31", "女");
return Arrays.asList(entity1, entity2);
} }

然后在消费者端通过修改Reference注解的version属性来控制调用:

@RestController
public class UserController {
@Reference(check = false, retries=2, version = "v1")
private UserService userService; @GetMapping("/selectUser")
public List<UserEntity> selectUse() {
return userService.selectList();
}
}

启动当前消费者,然后将v1修改为v2,修改端口号来启动第二个消费者,分别调用两个接口,结果为:

Stub本地存根

本地存根,指的是除开远程调用的实现之外,自己本地也有实现,且可以根据参数来决定是否使用本地实现

在消费者端创建一个接口的实现类:

// 这个实现类就是本地存根类
public class UserServiceImpl implements UserService {
// 这里用来存放远程调用的service对象
private final UserService userService;
// 远程对象通过构造方法传入
public UserServiceImpl(UserService userService) {
this.userService = userService;
}
// 复写方法中根据各种情况来判断是否使用远程实例
@Override
public List<UserEntity> selectList() {
try {
return userService.selectList();
} catch (Exception e) {
System.err.println("远程调用出现异常!");
return null;
}
} }

然后在controller位置的@Reference注解使用stub属性,属性值为本地存根的类的全路径名:

@RestController
public class UserController { @Reference(check = false, stub = "site.hanzhe.service.impl.UserServiceImpl")
private UserService userService; @GetMapping("/selectUser")
public List<UserEntity> selectUse() {
return userService.selectList();
} }

可以在服务提供者的位置模拟异常来查看是否设置成功

Dubbo学习笔记(二) Dubbo的基本配置的更多相关文章

  1. dubbo学习笔记(二)dubbo中的filter

    转:https://www.cnblogs.com/cdfive2018/p/10219730.html dubbo框架提供了filter机制的扩展点(本文基于dubbo2.6.0版本). 扩展接口 ...

  2. dubbo学习笔记二(服务调用)

    项目结构 代码示例 由于之前的IEchoService 的一个方法只是在服务端控制台打印,不便在浏览器测试,所以新添加的方法 api和服务端代码变更 public interface IEchoSer ...

  3. Hibernate学习笔记二:常用映射配置

    转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/6760895.html 一:单向一对一 常用唯一外键的方法来配置单向一对一关系. 1:实体关系 类A中有类B对象 ...

  4. JMX学习笔记(二)-Notification

    Notification通知,也可理解为消息,有通知,必然有发送通知的广播,JMX这里采用了一种订阅的方式,类似于观察者模式,注册一个观察者到广播里,当有通知时,广播通过调用观察者,逐一通知. 这里写 ...

  5. InterSystems Ensemble学习笔记(二) Ensemble创建镜像, 实现自动故障转移

    系列目录 InterSystems Ensemble学习笔记(一) Ensemble介绍及安装InterSystems Ensemble学习笔记(二) Ensemble创建镜像, 实现自动故障转移 一 ...

  6. kvm虚拟化学习笔记(二)之linux kvm虚拟机安装

    KVM虚拟化学习笔记系列文章列表----------------------------------------kvm虚拟化学习笔记(一)之kvm虚拟化环境安装http://koumm.blog.51 ...

  7. amazeui学习笔记二(进阶开发5)--Web 组件开发规范Rules

    amazeui学习笔记二(进阶开发5)--Web 组件开发规范Rules 一.总结 1.见名知意:见那些class名字知意,见函数名知意,见文件名知意 例如(HISTORY.md Web 组件更新历史 ...

  8. JDBC学习笔记二

    JDBC学习笔记二 4.execute()方法执行SQL语句 execute几乎可以执行任何SQL语句,当execute执行过SQL语句之后会返回一个布尔类型的值,代表是否返回了ResultSet对象 ...

  9. ZooKeeper学习笔记二:API基本使用

    Grey ZooKeeper学习笔记二:API基本使用 准备工作 搭建一个zk集群,参考ZooKeeper学习笔记一:集群搭建. 确保项目可以访问集群的每个节点 新建一个基于jdk1.8的maven项 ...

随机推荐

  1. setScaledContents

    ui->catchPhotoLabel_607->setPixmap(QPixmap::fromImage(*m_imageCatchtDefaultPhoto_607).scaled(Q ...

  2. 缓存cache和缓冲区buffer

    一.cache 1.cache的定义.从宏观上讲,缓存是处理速度不匹配的问题.可以是静态缓存(内存缓存.磁盘缓存).动态缓存(前端的缓存)和数据库缓存.另一个角度,从CPU来看,可以是寄存器和内存之间 ...

  3. 全局解决Vue跳转相同路由导致报错的问题

    大家使用Vue做开发的时候应该都遇到过这个问题,就是某个页面下调用this.$router.push(path),而path指向的页面和当前页面是同一页面时,就会发生报错,vue-router会提示你 ...

  4. CentOS 7关闭firewalld启用iptables 开放端口

    在CentOS7中,有很多CentOS 6中的常用服务发生了变化. 其中iptables是其中比较大的一个.防火墙iptables被firewalld取代. 本文将介绍,如果采用systemctl关闭 ...

  5. pytorch(17)学习率调整

    学习率调整 class _LRScheduler 主要属性 optimizer:关联的优化器 last_epoch:记录epoch数 bash_lrs:记录初始学习率 class _LRSchedul ...

  6. java下载文件指定目录下的文件

    方法一: @RequestMapping('download')def download(HttpServletRequest request, HttpServletResponse respons ...

  7. net5 中 后台任务利器之Hangfire 的使用

    什么是Hangfire Hangfire 是一个开源的.NET任务调度框架,目前1.6+版本已支持.NET Core.它最大特点在于内置提供集成化的控制台,方便后台查看及监控: 另外,Hangfire ...

  8. Vulkan移植GpuImage(一)高斯模糊与自适应阈值

    自适应阈值效果图 demo 这几天抽空看了下GpuImage的filter,移植了高斯模糊与自适应阈值的vulkan compute shader实现,一个是基本的图像处理,一个是组合基础图像处理聚合 ...

  9. Python3中变量作用域nonlocal的总结

    最近,在工作中踩到了一个关于Python3中nonlocal语句指定的变量作用域的坑.今天趁周六休息总结记录一下. 众所周知,Python中最常见的作用域定义如下:   但是,为了更加方便地在闭包函数 ...

  10. 基础篇:java.security框架之签名、加密、摘要及证书

    前言 和前端进行数据交互时或者和第三方商家对接时,需要对隐私数据进行加密.单向加密,对称加密,非对称加密,其对应的算法也各式各样.java提供了统一的框架来规范(java.security)安全加密这 ...