Java Dubbo服务调用扩展点Filter使用教程

 更新时间:2022年12月09日 11:10:52   作者:天使中的魔鬼M  
Dubbo是阿里巴巴公司开源的一个高性能优秀的服务框架,使得应用可通过高性能的RPC实现服务的输出和输入功能,可以和Spring框架无缝集成

扩展点介绍

如上图所示,从服务调用的角度来看,Dubbo 在链路中提供了丰富的扩展点,覆盖了负载均衡方式、选址前后的拦截器、服务端处理拦截器等。 简单来说 Dubbo 发起远程调用的时候,主要工作流程可以分为消费端和服务端两个部分。

消费端的工作流程如下:

通过 Stub 接收来自用户的请求,并且封装在 Invocation 对象中

将 Invocation 对象传递给 ClusterFilter(扩展点)做选址前的请求预处理,如请求参数的转换、请求日志记录、限流等操作都是在此阶段进行的

将 Invocation 对象传递给 Cluster(扩展点)进行集群调用逻辑的决策,如快速失败模式、安全失败模式等决策都是在此阶段进行的

  • Cluster 调用 Directory 获取所有可用的服务端地址信息
  • Directory 调用 StateRouter(扩展点,推荐使用) 和 Router(扩展点) 对服务端的地址信息进行路由筛选,此阶段主要是从全量的地址信息中筛选出本次调用允许调用到的目标,如基于打标的流量路由就是在此阶段进行的
  • Cluster 获得从 Directory 提供的可用服务端信息后,会调用 LoadBalance (扩展点)从多个地址中选择出一个本次调用的目标,如随机调用、轮询调用、一致性哈希等策略都是在此阶段进行的
  • Cluster 获得目标的 Invoker 以后将 Invocation 传递给对应的 Invoker,并等待返回结果,如果出现报错则执行对应的决策(如快速失败、安全失败等)

经过上面的处理,得到了带有目标地址信息的 Invoker,会再调用 Filter(扩展点)进行选址后的请求处理(由于在消费端侧创建的 Filter 数量级和服务端地址量级一致,如无特殊需要建议使用 ClusterFilter 进行扩展拦截,以提高性能)

最后 Invocation 会被通过网络发送给服务端

服务端的工作流程如下:

服务端通信层收到请求以后,会将请求传递给协议层构建出 Invocation

将 Invocation 对象传递给 Filter (扩展点)做服务端请求的预处理,如服务端鉴权、日志记录、限流等操作都是在此阶段进行的

将 Invocation 对象传递给动态代理做真实的服务端调用

介绍完扩展点后,下面以拦截扩展点为例介绍具体使用

拦截点

官方接口: org.apache.dubbo.rpc.Filter

@SPI(scope = ExtensionScope.MODULE)
public interface Filter extends BaseFilter {
}

在项目中实现该接口,自定义逻辑:

@Slf4j
@Activate(group = CommonConstants.CONSUMER)
public class ParamFilter implements Filter {
    @Override
    public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException {
		// 这里可以获取到消费者请求的相关信息,如服务名、请求的方法名、以及请求的参数
		// 实际应用:做一些参数检查以及转换等等操作
        log.info("Service-Name: " + invocation.getServiceName() + " Method: " + invocation.getMethodName() + " Parameter: " + Arrays.toString(invocation.getArguments()));
        // 将请求发送给服务端
        return invoker.invoke(invocation);
    }
}

编写Filter的配置文件:

文件名:org.apache.dubbo.rpc.Filter

内容:

MyFilter=com.mxf.filter.ParamFilter

MyFilter: 是自定义Filter的一个别名,可以自定义

在application.yaml中配置该接口:

dubbo:
  application:
    name: shop-service-consumer
  protocol:
    name: dubbo
    port: -1
  registry:
    id: nacos-registry
    address: nacos://192.168.11.233:8848
  config-center:
    address: nacos://192.168.11.233:8848
  metadata-report:
    address: nacos://192.168.11.233:8848
  consumer:			  #在消费端的拦截扩展点
    filter: MyFilter  # 注意名称要和签名定义的一致

重启项目演示

可以使用Postman请求相关接口,就可以看到输出日志:

2022-11-23 11:39:36.463  INFO 8988 --- [nio-8082-exec-1] com.mxf.filter.ParamFilter               : Service-Name: com.mxf.service.ShopService Method: queryByName Parameter: [xiao]

可以看到在消费端自定义的拦截扩展点已经生效,可以获取到相关信息

Dubbo实现负载均衡的扩展

官方提供的接口:

RandomLoadBalance:随机策略的LB

RoundRobinLoadBalance:轮询策略的LB

LeastActiveLoadBalance:最少活跃调用数策略的LB:举例有两台服务器,一台正在处理的任务数很多也就是负载高,那么当前负载策略就会尽量少的给这台服务器分配任务;反之,如果一台服务器处理的任务数少,就尽量优先给它分配任务。

ConsistentHashLoadBalance:一致性Hash策略的LB:。一致性Hash,相同参数的请求总是发到同一提供者,当某一台提供者挂时,原本发往该提供者的请求,基于虚拟节点,平摊到其它提供者,不会引起剧烈变动。

ShortestResponseLoadBalance:最短响应优先策略的LB

除了以上Dubbo提供的五种默认实现的负载均衡策略,也可以自定义负载均衡策略,只要实现Dubbo提供的LoadBalance接口,自定义实现select即可:

@SPI(RandomLoadBalance.NAME)
public interface LoadBalance {
    /**
     * select one invoker in list.
     *
     * @param invokers   invokers.
     * @param url        refer url
     * @param invocation invocation.
     * @return selected invoker.
     */
    @Adaptive("loadbalance")
    <T> Invoker<T> select(List<Invoker<T>> invokers, URL url, Invocation invocation) throws RpcException;
}

用法:以轮询策略为例

第一步编写配置文件:

文件名:org.apache.dubbo.rpc.cluster.LoadBalance

RRLB=org.apache.dubbo.rpc.cluster.loadbalance.RoundRobinLoadBalance

RRLB是自定义的负载均衡名称

在application.yaml中配置LB:

dubbo:
  consumer:	# 在消费端的LB
    loadbalance: RRLB

OK,到此配置完成,当启动多个服务提供者,用一个消费者去调用,就是采取轮询的策略均匀请求到服务提供者上

其余四种实现方式一样。

到此这篇关于Java Dubbo服务调用扩展点Filter使用教程的文章就介绍到这了,更多相关Java Filter内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 基于java web获取网页访问次数代码实例

    基于java web获取网页访问次数代码实例

    这篇文章主要介绍了基于java web获取网页访问次数代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-02-02
  • Spring中BeanFactory解析bean详解

    Spring中BeanFactory解析bean详解

    本篇文章主要介绍了Spring中BeanFactory解析bean详解 ,详细的介绍了使用BeanFactory对bean进行解析的实例,有兴趣的可以了解一下。
    2017-04-04
  • Java利用TreeUtils工具类实现列表转树

    Java利用TreeUtils工具类实现列表转树

    在开发过程中,总有列表转树的需求,几乎是项目的标配,有没有一种通用且跨项目的解决方式呢?本文将基于Java8的Lambda 表达式和Stream等知识,使用TreeUtils工具类实现一行代码完成列表转树这一通用型需求,需要的可以参考一下
    2022-11-11
  • Java实现pdf文件合并的使用示例

    Java实现pdf文件合并的使用示例

    本文主要介绍了Java实现pdf文件合并的使用示例,主要是将需要合并的pdf文件都拷贝到指定目录a中,调用该工具类将该目录作为第一个参数,第二个参数传入输出文件对象即可,感兴趣的可以了解一下
    2023-12-12
  • Java Stream流使用最多的方式示例详解

    Java Stream流使用最多的方式示例详解

    在 Java 编程中,Stream 流提供了一种高效、便捷的方式来处理集合数据,本文将详细介绍 Java 中 Stream 流的用法,包括基础用法、中级用法、高级用法以及一些特殊方法的使用,感兴趣的朋友一起看看吧
    2024-12-12
  • mybatis通过XML的方式拼接动态sql

    mybatis通过XML的方式拼接动态sql

    动态SQL是一种在运行时构造和执行SQL语句的技术,这篇文章主要为大家介绍了mybatis如何通过XML的方式拼接动态sql,有需要的小伙伴可以参考一下
    2024-12-12
  • 详解SpringBoot多跨域请求的支持(JSONP)

    详解SpringBoot多跨域请求的支持(JSONP)

    跨域是很多项目需要遇到的文章,本篇文章主要介绍了详解SpringBoot多跨域请求的支持(JSONP),具有一定的参考价值,有兴趣的可以了解一下
    2017-04-04
  • java实现外卖订餐系统

    java实现外卖订餐系统

    这篇文章主要为大家详细介绍了java实现外卖订餐系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-01-01
  • SpringBoot  jdbctemplate使用方法解析

    SpringBoot jdbctemplate使用方法解析

    这篇文章主要介绍了SpringBoot jdbctemplate使用方法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-05-05
  • 解析Spring Boot 如何让你的 bean 在其他 bean 之前完成加载

    解析Spring Boot 如何让你的 bean 在其他 bean&n

    在 SpringBoot 中如何让自己的某个指定的 Bean 在其他 Bean 前完成被 Spring 加载?我听到这个问题的第一反应是,为什么会有这样奇怪的需求?下面小编给大家分析下Spring Boot 如何让你的 bean 在其他 bean 之前完成加载 ,感兴趣的朋友一起看看吧
    2024-01-01

最新评论