在现代微服务架构中,服务之间的通信与协作是非常重要的。Spring Cloud Alibaba 提供了一套完整的微服务解决方案,其中包括 Nacos 用于服务注册与发现,OpenFeign 用于声明式服务调用,Spring Cloud LoadBalancer 用于负载均衡。本文将通过一个简单的电商系统示例,演示如何使用这些组件来实现服务之间的交互。
在本示例中,我们将实现一个电商系统,包含以下三个服务:
每个服务将注册到 Nacos 服务注册中心,并通过 OpenFeign 实现服务之间的调用。此外,我们将使用 Spring Cloud LoadBalancer 来实现负载均衡。
在开始之前,确保以下环境和依赖已安装:
所有服务都将注册到 Nacos,因此需要在每个服务的 application.yml
中配置 Nacos。
spring:
cloud:
nacos:
discovery:
server-addr: localhost:8848 # Nacos 服务地址
商品服务提供商品信息查询接口。
product-service
├── src
│ ├── main
│ │ ├── java
│ │ │ └── com.example.product
│ │ │ ├── ProductServiceApplication.java
│ │ │ ├── controller
│ │ │ │ └── ProductController.java
│ │ │ └── model
│ │ │ └── Product.java
│ │ └── resources
│ │ └── application.yml
└── pom.xml
ProductServiceApplication.java
package com.example.product;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
@SpringBootApplication
@EnableDiscoveryClient // 启用服务发现
public class ProductServiceApplication {
public static void main(String[] args) {
SpringApplication.run(ProductServiceApplication.class, args);
}
}
ProductController.java
package com.example.product.controller;
import com.example.product.model.Product;
import org.springframework.web.bind.annotation.*;
@RestController
@RequestMapping("/api/products")
public class ProductController {
@GetMapping("/{id}")
public Product getProductById(@PathVariable Long id) {
// 模拟从数据库获取商品信息
return new Product(id, "Sample Product", "This is a sample product", 99.99);
}
}
Product.java
package com.example.product.model;
public class Product {
private Long id;
private String name;
private String description;
private Double price;
public Product(Long id, String name, String description, Double price) {
this.id = id;
this.name = name;
this.description = description;
this.price = price;
}
// Getter 和 Setter 方法
// ...
}
application.yml
server:
port: 8081
spring:
application:
name: product-service
cloud:
nacos:
discovery:
server-addr: localhost:8848
用户服务提供用户信息管理和验证接口。
user-service
├── src
│ ├── main
│ │ ├── java
│ │ │ └── com.example.user
│ │ │ ├── UserServiceApplication.java
│ │ │ ├── controller
│ │ │ │ └── UserController.java
│ │ │ └── model
│ │ │ └── User.java
│ │ └── resources
│ │ └── application.yml
└── pom.xml
UserServiceApplication.java
package com.example.user;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
@SpringBootApplication
@EnableDiscoveryClient // 启用服务发现
public class UserServiceApplication {
public static void main(String[] args) {
SpringApplication.run(UserServiceApplication.class, args);
}
}
UserController.java
package com.example.user.controller;
import com.example.user.model.User;
import org.springframework.web.bind.annotation.*;
@RestController
@RequestMapping("/api/users")
public class UserController {
@GetMapping("/{id}")
public User getUserById(@PathVariable Long id) {
// 模拟从数据库获取用户信息
return new User(id, "John Doe", "[email protected]");
}
}
User.java
package com.example.user.model;
public class User {
private Long id;
private String username;
private String email;
public User(Long id, String username, String email) {
this.id = id;
this.username = username;
this.email = email;
}
// Getter 和 Setter 方法
// ...
}
application.yml
server:
port: 8082
spring:
application:
name: user-service
cloud:
nacos:
discovery:
server-addr: localhost:8848
订单服务将调用商品服务和用户服务来创建订单。
order-service
├── src
│ ├── main
│ │ ├── java
│ │ │ └── com.example.order
│ │ │ ├── OrderServiceApplication.java
│ │ │ ├── controller
│ │ │ │ └── OrderController.java
│ │ │ ├── feign
│ │ │ │ ├── ProductClient.java
│ │ │ │ └── UserClient.java
│ │ │ └── model
│ │ │ ├── Order.java
│ │ │ ├── OrderRequest.java
│ │ │ ├── Product.java
│ │ │ └── User.java
│ │ └── resources
│ │ └── application.yml
└── pom.xml
OrderServiceApplication.java
package com.example.order;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import org.springframework.cloud.openfeign.EnableFeignClients;
@SpringBootApplication
@EnableDiscoveryClient // 启用服务发现
@EnableFeignClients // 启用 OpenFeign 客户端
public class OrderServiceApplication {
public static void main(String[] args) {
SpringApplication.run(OrderServiceApplication.class, args);
}
}
OrderController.java
package com.example.order.controller;
import com.example.order.feign.ProductClient;
import com.example.order.feign.UserClient;
import com.example.order.model.Order;
import com.example.order.model.OrderRequest;
import com.example.order.model.Product;
import com.example.order.model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
@RestController
@RequestMapping("/api/orders")
public class OrderController {
@Autowired
private ProductClient productClient;
@Autowired
private UserClient userClient;
@PostMapping
public Order createOrder(@RequestBody OrderRequest orderRequest) {
// 通过 OpenFeign 调用商品服务
Product product = productClient.getProductById(orderRequest.getProductId());
// 通过 OpenFeign 调用用户服务
User user = userClient.getUserById(orderRequest.getUserId());
// 创建订单逻辑
return new Order(user.getId(), product.getId(), 1, product.getPrice());
}
}
ProductClient.java
package com.example.order.feign;
import com.example.order.model.Product;
import org.springframework
.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
@FeignClient(name = "product-service")
public interface ProductClient {
@GetMapping("/api/products/{id}")
Product getProductById(@PathVariable("id") Long id);
}
UserClient.java
package com.example.order.feign;
import com.example.order.model.User;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
@FeignClient(name = "user-service")
public interface UserClient {
@GetMapping("/api/users/{id}")
User getUserById(@PathVariable("id") Long id);
}
Order.java
package com.example.order.model;
public class Order {
private Long userId;
private Long productId;
private Integer quantity;
private Double totalPrice;
public Order(Long userId, Long productId, Integer quantity, Double totalPrice) {
this.userId = userId;
this.productId = productId;
this.quantity = quantity;
this.totalPrice = totalPrice;
}
// Getter 和 Setter 方法
// ...
}
OrderRequest.java
package com.example.order.model;
public class OrderRequest {
private Long userId;
private Long productId;
public OrderRequest(Long userId, Long productId) {
this.userId = userId;
this.productId = productId;
}
// Getter 和 Setter 方法
// ...
}
application.yml
server:
port: 8083
spring:
application:
name: order-service
cloud:
nacos:
discovery:
server-addr: localhost:8848
loadbalancer:
ribbon:
enabled: false # 使用 Spring Cloud LoadBalancer 代替 Ribbon
启动所有三个服务,确保它们正确注册到 Nacos。在 Nacos 控制台中,你应该能够看到 order-service
、product-service
和 user-service
的实例。
order-service
的 /api/orders
接口,传入 OrderRequest
数据。order-service
将通过 ProductClient
和 UserClient
调用相应的商品服务和用户服务,获取商品和用户信息,并创建订单。本文通过一个完整的电商系统示例,展示了如何使用 Spring Cloud Alibaba Nacos 进行服务注册与发现,使用 OpenFeign 进行声明式服务调用,以及使用 Spring Cloud LoadBalancer 实现负载均衡。通过 Nacos、OpenFeign 和 LoadBalancer 的协同工作,我们能够轻松构建一个稳定、可靠且可扩展的微服务架构。希望本文的示例能够为你在实际项目中应用这些技术提供参考。