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. Django模型层2

    目录 一.聚合查询 聚合函数 二.分组查询 利用group by进行分组查询 三.F与Q查询 1. F类 2. Q类 四.orm字段及参数 五.自定义char字段 六.orm中的事务操作 1. 什么是 ...

  2. Gradle 差异化构建

    Compile 默认的依赖方式,任何情况下都会依赖. Provided 只提供编译时依赖,打包时不会添加进去. Apk 只在打包Apk包时依赖,这个应该是比较少用到的. TestCompile 只在测 ...

  3. docker封装nuxt项目使用jenkins发布

    一.概述 vue项目可以打一个dist静态资源包,直接使用Nginx发布即可. 但是nuxt项目无法像vue那样,可以打一个dist静态资源包. 需要安装Node.js,并使用npm install ...

  4. python进阶(12)闭包

    闭包 首先了解一下:如果在一个函数的内部定义了另一个函数,外部的我们叫他外函数,内部的我们叫他内函数. 在一个外函数中定义了一个内函数,内函数里运用了外函数的临时变量,并且外函数的返回值是内函数的引用 ...

  5. MySQL:安装与配置

    记录一次 MySQL 在Windows系统的安装配置过程 安装MySQL 0.下载社区版安装包 官网下载地址:https://dev.mysql.com/downloads/installer/ 1. ...

  6. Django 自定义标签与过滤器报错 No module named 'templatetags'

    Django 自定义标签与过滤器报错 按照网上的教程如果想使用自定义的标签与过滤器就得往settings.py中添加下列数据 TEMPLATES = [ { 'BACKEND': 'django.te ...

  7. python中lambda、yield、map、filter、reduce的使用

    1. 匿名函数lambda python中允许使用lambda关键字定义一个匿名函数.所谓的匿名函数就是说使用一次或者几次之后就不再需要的函数,属于"一次性"函数. #例1:求两数 ...

  8. 【图像处理】使用OpenCV+Python进行图像处理入门教程(二)

    这篇随笔介绍使用OpenCV进行图像处理的第二章 图像的运算,让我们踏上继续回顾OpenCV进行图像处理的奇妙之旅,不断地总结.回顾,以新的视角快速融入计算机视觉的奥秘世界. 2  图像的运算 复杂的 ...

  9. Node.js 模块化你所需要知道的事

    一.前言 我们知道,Node.js是基于CommonJS规范进行模块化管理的,模块化是面对复杂的业务场景不可或缺的工具,或许你经常使用它,但却从没有系统的了解过,所以今天我们来聊一聊Node.js模块 ...

  10. 推荐!!! Markdown图标索引网站

    作者:三十三重天 博客: http://www.zhouhuibo.club 我们在观察别人的文章时候时,总能看到很多有趣的图标,像是这样