乐趣区

关于阿里云:微服务最佳实践零改造实现-Spring-Cloud-Apache-Dubbo-互通

作者:孙彩荣

很遗憾,这不是一篇对于中间件实践或原理解说的文章,没有浅近艰涩的工作原理剖析,文后也没有令人惊叹的工程数字统计。 本文以理论我的项目和代码为示例,一步一步演示如何以最低老本实现 Apache Dubbo 体系与 Spring Cloud 体系的互通,进而实现不同微服务体系的混合部署、迁徙等,帮忙您解决理论架构及业务问题。

背景与指标

如果你在微服务开发过程中正面临以下一些业务场景须要解决,那么这篇文章能够帮到您:

  • 您曾经有一套基于 Dubbo 构建的微服务利用,这时你须要将局部服务通过 REST HTTP 的模式(非接口、办法模式)公布进来,供一些规范的 HTTP 端调用(如 Spring Cloud 客户端),整个过程最好是不必改代码,间接为写好的 Dubbo 服务加一些配置、注解就能实现。
  • 您曾经有一套基于 Spring Cloud 构建的微服务体系,而后又构建了一套 Dubbo 体系的微服务,你想两套体系共存,因而当初两边都须要调用到对方公布的服务。也就是 Dubbo 利用作为生产方要调用到 Spring Cloud 公布的 HTTP 接口,Dubbo 利用作为提供方还能公布 HTTP 接口给 Spring Cloud 调用。
  • 出于一些历史起因,你正布局从一个微服务体系迁徙到另外一个微服务体系,前提条件是要保障两头过程的平滑迁徙。

对于以上几个场景,咱们都能够借助 Dubbo3 内置的 REST 编程范式反对实现,这让 Dubbo 既能够作为生产方调用 HTTP 接口的服务,又能够作为提供方对外公布 REST 格调的 HTTP 服务,同时整个编码过程反对业界罕用的 REST 编程范式(如 JAX-RS、Spring MVC 等),因而能够做到根本不改变任何代码的状况下实现 Dubbo 与 Spring Cloud 体系的相互调用。

  • 对于这一部分更多的设计与实践论述请参见这里的博客文章 [ 1]
  • 对于 Dubbo REST 的更多配置形式请参见 rest 应用参考手册 [ 2]

示例一:Dubbo 调用 Spring Cloud

在曾经有一套 Spring Cloud 微服务体系的状况下,演示如何应用 Dubbo 调用 Spring Cloud 服务(蕴含主动的地址发现与协定传输)。在注册核心方面,本示例应用 Nacos 作为注册核心,对于 Zookeeper、Consul 等两种体系都反对的注册核心同样实用。

构想你曾经有一套 Spring Cloud 的微服务体系,当初咱们将引入 Dubbo 框架,让 Dubbo 利用可能失常的调用到 Spring Cloud 公布的服务。本示例残缺源码请参见 samples/dubbo-call-sc [ 3]

启动 Spring Cloud Server

示例中 Spring Cloud 利用的构造如下:

利用配置文件如下:

server:
  port: 8099
spring:
  application:
    name: spring-cloud-provider-for-dubbo
  cloud:
    nacos:
      serverAddr: 127.0.0.1:8848 #注册核心 

以下是一个非常简单的 Controller 定义,公布了一个 /users/list/ 的 http 端点。

@RestController
@RequestMapping("/users")
public class UserController {@GetMapping("/list")
    public List<User> getUser() {return Collections.singletonList(new User(1L, "spring cloud server"));
    }
}

启动 SpringCloudApplication,通过 cURL 或浏览器拜访 http://localhost:8099/users/list 能够测试利用启动胜利。

应用 Dubbo Client 调用服务

Dubbo client 也是一个规范的 Dubbo 利用,我的项目根本构造如下:

其中,一个比拟要害的是如下接口定义(失常状况下,以下接口能够间接从原有的 Spring Cloud client 利用中原样拷贝过去即可,无需做任何批改)。

如果之前没有基于 OpenFeign 的 Spring Cloud 生产端利用,那么就须要自行定义一个接口,此时不肯定要应用 OpenFeign 注解,应用 Spring MVC 规范注解即可。

通过 DubboReference 注解将 UserServiceFeign 接口注册为 Dubbo 服务。

@DubboReference
private UserServiceFeign userService;

接下来,咱们就能够用 Dubbo 规范的形式对服务发动调用了。

List<User> users = userService.users();

通过 DubboConsumerApplication 启动 Dubbo 利用,验证能够胜利调用到 Spring Cloud 服务。

示例二:Spring Cloud 调用 Dubbo

在接下来的示例中,咱们将展现如何将 Dubbo server 公布的服务凋谢给 Spring Cloud client 调用。

示例的相干源码在 samples/sc-call-dubbo [ 4]

启动 Dubbo Server

Dubbo server 利用的代码构造非常简单,是一个典型的 Dubbo 利用。

相比于一般的 Dubbo 服务定义,咱们要在接口上加上如下规范 Spring MVC 注解:

@RestController
@RequestMapping("/users")
public interface UserService {@GetMapping(value = "/list")
    List<User> getUsers();}

除了以上注解之外,其余服务公布等流程都统一,应用 DubboService 注解公布服务即可:

@DubboService
public class UserServiceImpl implements UserService {
    @Override
    public List<User> getUsers() {return Collections.singletonList(new User(1L, "Dubbo provider!"));
    }
}

在服务配置上,特地留神咱们须要将服务的协定配置为 rest protocol: rest,地址发现模式应用 register-mode: instance:

dubbo:
  registry:
    address: nacos://127.0.0.1:8848
    register-mode: instance
  protocol:
    name: rest
    port: 8090

启动 Dubbo 利用,此时拜访以下地址能够验证服务运行失常:http://localhost:8090/users/list

应用 Spring Cloud 调用 Dubbo

应用 OpenFeign 开发一个规范的 Spring Cloud 利用,即可调用以上公布的 Dubbo 服务,我的项目代码构造如下:

其中,咱们定义了一个 OpenFeign 接口,用于调用下面公布的 Dubbo rest 服务。

@FeignClient(name = "dubbo-provider-for-spring-cloud")
public interface UserServiceFeign {@RequestMapping(value = "/users/list", method = RequestMethod.GET)
    List<User> getUsers();}

定义以下 controller 作为 OpenFeign 和 RestTemplate 测试入口:

public class UserController {

    private final RestTemplate restTemplate;
    private final UserServiceFeign userServiceFeign;

    public UserController(RestTemplate restTemplate,
                          UserServiceFeign userServiceFeign) {
        this.restTemplate = restTemplate;
        this.userServiceFeign = userServiceFeign;
    }

    @RequestMapping("/rest/test1")
    public String doRestAliveUsingEurekaAndRibbon() {
        String url = "http://dubbo-provider-for-spring-cloud/users/list";
        System.out.println("url:" + url);
        return restTemplate.getForObject(url, String.class);
    }

    @RequestMapping("/rest/test2")
    public List<User> doRestAliveUsingFeign() {return userServiceFeign.getUsers();
    }
}

依据以上 Controller 定义,咱们能够别离拜访以下地址进行验证:

  • OpenFeign 形式: http://localhost:8099/dubbo/rest/test1
  • RestTemplage 形式: http://localhost:8099/dubbo/rest/test2

为 Dubbo Server 公布更多的服务

咱们能够利用 Dubbo 的多协定公布机制,为一些服务配置多协定公布。接下来,咱们就为下面提到的 Dubbo server 服务减少 dubbo tcp 协定公布,从而达到以下部署成果,让这个 Dubbo 利用同时服务 Dubbo 微服务体系和 Spring Cloud 微服务体系。

为了实现这个成果,咱们只须要在配置中减少多协定配置即可:

dubbo:
  protocols:
    - id: rest
      name: rest
      port: 8090
    - id: dubbo
      name: dubbo
      port: 20880

同时,服务注解中也配置为多协定公布:

@DubboService(protocol="rest,dubbo")
public class UserServiceImpl implements UserService {}

这样,咱们就胜利的将 UserService 服务以 dubbo 和 rest 两种协定公布了进来(多端口多协定的形式),dubbo 协定为 Dubbo 体系服务,rest 协定为 Spring Cloud 体系服务。

留神: Dubbo 为多协定公布提供了单端口、多端口两种形式,这样的灵活性对于不同部署环境下的服务会有比拟大的帮忙。在确定您须要的多协定公布形式前,请提仔细阅读以下多协定配置 [ 5] 文档。

总结

基于 Dubbo 的 rest 编程范式、多协定公布等个性,能够帮忙你轻松的实现 Dubbo 服务的 http 协定公布,让后端服务基于 RPC 高效通信的同时,可能更容易的与 http 服务体系买通,本示例通过 Dubbo 与 Spring Cloud 两套体系的共存、互通示例十分清晰的演示了编码过程。

此局部内容的正式版本将在 Dubbo 3.3.0 版本正式公布,同时还蕴含 Triple 协定的重磅降级,敬请期待!

相干链接:

[1] 博客文章

https://cn.dubbo.apache.org/zh-cn/blog/2023/01/05/dubbo-%e8%bf%9e%e6%8e%a5%e5%bc%82%e6%9e%84%e5%be%ae%e6%9c%8d%e5%8a%a1%e4%bd%93%e7%b3%bb-%e5%a4%9a%e5%8d%8f%e8%ae%ae%e5%a4%9a%e6%b3%a8%e5%86%8c%e4%b8%ad%e5%bf%83/

[2] rest 应用参考手册

https://cn.dubbo.apache.org/zh-cn/overview/reference/proposals/protocol-http/

[3] samples/dubbo-call-sc

https://github.com/apache/dubbo-samples/tree/master/2-advanced/dubbo-samples-springcloud/dubbo-call-sc

[4] samples/sc-call-dubbo

https://github.com/apache/dubbo-samples/tree/master/2-advanced/dubbo-samples-springcloud/sc-call-dubbo

[5] 多协定配置

https://cn.dubbo.apache.org/zh-cn/overview/mannual/java-sdk/advanced-features-and-usage/service/multi-protocols/

退出移动版