在微服务架构中,数据孤岛是一个常见的问题。当我们进行业务开发时,经常会遇到这样的场景:一个业务实体(如订单)需要展示另一个业务实体(如用户)的信息。今天,我们就通过一个经典的"订单查用户"案例,深入剖析如何使用 RestTemplate 实现微服务间的远程调用与数据组装。
一、需求背景:当订单遇上用户
假设我们正在开发一个商城系统,目前已经将"订单服务"(Order Service)和"用户服务"(User Service)进行了拆分,并且遵循微服务的数据库隔离原则------订单库只存订单数据,用户库只存用户数据。
当前的业务需求 :
前端页面需要展示订单详情,且不仅要显示商品名称和价格,还要显示下单用户的姓名和地址。
遇到的问题 :
当我们调用订单服务的查询接口 GET /order/101 时,返回的 JSON 数据中,用户信息字段(user)是 null。这是因为订单表虽然存了 userId,但它并没有直接持有用户的详细信息(如姓名、地址),这些信息存储在独立的用户服务数据库中。
二、核心思路:HTTP 远程调用
为了解决这个问题,我们需要打破服务间的壁垒,但又不能违背微服务的初衷。
不可行的方案:直接查库
千万不要在订单服务中配置用户数据库的连接信息,直接通过 SQL 去查询用户表。这种做法被称为"跨库查询",它会导致服务间产生强耦合,一旦用户库表结构变更,订单服务也会受影响,完全违背了微服务"高内聚、低耦合"的设计原则。
可行的方案:远程调用
正确的做法是"面向接口编程"。既然用户服务已经提供了查询用户的 RESTful 接口(例如 GET /user/{id}),订单服务只需要像浏览器一样,向这个接口发起一个 HTTP 请求,获取 JSON 数据,然后将其组装到订单对象中即可。
三、实战步骤:三步实现远程调用
在 Spring 生态中,实现 HTTP 请求最基础且强大的工具就是 RestTemplate。我们将通过以下三个步骤完成功能。
1. 注册 RestTemplate
RestTemplate 是一个 Spring 提供的用于发送 HTTP 请求的工具类。为了在业务代码中方便地使用它,我们需要将其注册为 Spring 容器中的一个 Bean。
通常,我们会在订单服务的启动类(OrderApplication)中完成这一步:
@SpringBootApplication
public class OrderApplication {
public static void main(String[] args) {
SpringApplication.run(OrderApplication.class, args);
}
// 将 RestTemplate 注册到 Spring 容器中
@Bean
public RestTemplate restTemplate() {
return new RestTemplate();
}
}
通过 @Bean 注解,Spring 会在启动时自动创建一个 RestTemplate 实例,我们可以随时通过 @Autowired 将其注入到任何地方。
2. 改造业务逻辑
接下来,我们需要修改订单服务的业务层(OrderService)。原本的逻辑只查询了本地数据库,现在我们需要增加远程调用的步骤。
核心代码实现:
@Service
public class OrderService {
@Autowired
private OrderMapper orderMapper;
@Autowired
private RestTemplate restTemplate; // 注入 RestTemplate
public Order queryOrderById(Long orderId) {
// 第一步:查询本地订单数据
Order order = orderMapper.findById(orderId);
// 第二步:远程调用用户服务,获取用户信息
// 1. 构建 URL,拼接用户 ID
String url = "http://localhost:8081/user/" + order.getUserId();
// 2. 发起 GET 请求,获取 User 对象
// getForObject 会自动将返回的 JSON 反序列化为 User 类
User user = restTemplate.getForObject(url, User.class);
// 第三步:组装数据
order.setUser(user);
// 第四步:返回组装后的完整订单
return order;
}
}
3. 数据组装与返回
通过上述代码,我们完成了一个完整的调用链:
- 查订单 :先根据 ID 从本地
tb_order表查出订单,此时userId已知。 - 调用户 :利用
RestTemplate向http://localhost:8081/user/{userId}发起请求。 - 转对象 :
RestTemplate接收到用户服务返回的 JSON 字符串(如{"id":1, "username":"张三", ...}),并利用 Jackson 库自动将其转换为 Java 的User对象。 - 合数据 :将
User对象设置到Order对象的user属性中。
最终,当前端再次访问订单接口时,就能拿到包含完整用户信息的订单数据了。
四、总结与思考
通过 RestTemplate 实现远程调用,是微服务开发中最基础的技能之一。它让我们能够以标准的 HTTP 协议为桥梁,将分散在不同服务中的数据灵活地组合在一起。
关键点回顾:
- 解耦:服务间通过 URL 交互,不共享数据库。
- 便捷 :
RestTemplate封装了繁琐的 HTTP 连接细节,一行代码即可完成调用和 JSON 转换。 - 灵活:这种方式与语言无关,只要对方提供 HTTP 接口,无论是 Java、Python 还是 Go 服务,都可以互相调用。
虽然
RestTemplate功能强大,但在复杂的微服务调用链中,硬编码 URL(如http://localhost:8081)会带来维护困难。在未来的学习中,我们将引入 OpenFeign 和 Nacos,进一步简化调用方式并实现动态的服务发现。