spring cloud实现各个微服务应用间的通信方式有多种,我们首先讲解RestTemplate来实现product服务和order服务之间的通信,product端看作server端,order作为调用端看作client端。
一.1.在product服务创建ServerController
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class ServerController {
@GetMapping("/msg")
public String msg(){
return "this is product msg 1";
}
}
2.在order服务创建ClientController
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;
@RestController
@Slf4j
public class ClientController {
//第二种方式
// @Autowired
// private LoadBalancerClient loadBalancerClient;
//第三种方式
@Autowired
private RestTemplate restTemplate;
@GetMapping("/getProductMsg")
public String getProductMsg() {
//1.第一种方式(直接使用restTemplate, url写死)
// RestTemplate restTemplate = new RestTemplate();
// String response = restTemplate.getForObject("http://localhost:8080/msg", String.class);
//2. 第二种方式(利用loadBalancerClient通过应用名获取url, 然后再使用restTemplate)
// RestTemplate restTemplate = new RestTemplate();
// ServiceInstance serviceInstance = loadBalancerClient.choose("PRODUCT");
// String url = String.format("http://%s:%s", serviceInstance.getHost(), serviceInstance.getPort()) + "/msg";
// String response = restTemplate.getForObject(url, String.class);
//3. 第三种方式(利用@LoadBalanced, 可在restTemplate里使用应用名字)
String response = restTemplate.getForObject("http://PRODUCT/msg", String.class);
log.info("response={}", response);
return response;
}
}
tips:在使用第三种方式时,需要写一个RestTemplate配置类,代码如下:
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
import org.springframework.web.client.RestTemplate;
@Component
public class RestTemplateConfig {
@Bean
@LoadBalanced
public RestTemplate restTemplate(){
return new RestTemplate();
}
}
二.接下来我们使用feign组件来实现应用间的通信(伪RPC),项目为多模块模式,分为server、client、common模块(重在思想)
1.product、order服务pom文件的修改:
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
2.在product的client端创建ProductClient接口,供order端调用(写完后要mvn clean install,否则order端引入不到)
import java.util.List;
@FeignClient(name="product")
@Component
public interface ProductClient {
/*
获取商品列表,给订单服务用的
*/
@PostMapping("/product/listForOrder")
List<ProductInfoOutput> listForOrder(@RequestBody List<String> productIdList);
@PostMapping("/product/decreaseStock")
void decreaseStock(@RequestBody List<DecreaseStockInput> decreaseStockInputList);
}
3.order服务的server模块引入product的client端,pom文件新增:
<dependency>
<groupId>com.imooc</groupId>
<artifactId>product-client</artifactId>
<version>${product-client.version}</version>
</dependency>
4.在order服务启动类上加注解@EnableFeignClients(basePackages = “com.xxx.product.client”)
5.在order 的service实现类上调用product的服务(附service关键代码)
import com.imooc.order.dataobject.OrderDetail;
import com.imooc.order.dataobject.OrderMaster;
import com.imooc.order.dto.CartDTO;
import com.imooc.order.dto.OrderDTO;
import com.imooc.order.enums.OrderStatusEnum;
import com.imooc.order.enums.PayStatusEnum;
import com.imooc.order.repository.OrderDetailRepository;
import com.imooc.order.repository.OrderMasterRepository;
import com.imooc.order.service.OrderService;
import com.imooc.order.utils.KeyUtil;
import com.imooc.product.client.ProductClient;
import com.imooc.product.common.DecreaseStockInput;
import com.imooc.product.common.ProductInfoOutput;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.List;
import java.util.stream.Collectors;
@Service
public class OrderServiceImpl implements OrderService {
@Autowired
private OrderMasterRepository orderMasterRepository;
@Autowired
private OrderDetailRepository orderDetailRepository;
@Autowired
private ProductClient productClient;
/*
2.查询商品信息(调用商品服务)
3.计算总价
4.扣库存(调用商品服务)
5.订单入库
*/
@Override
public OrderDTO create(OrderDTO orderDTO) {
String orderId=KeyUtil.genUniqueKey();
orderDTO.setOrderId(orderId);
//查询商品信息
List<String> productIdList=orderDTO.getOrderDetailList().stream().map(OrderDetail::getProductId).collect(Collectors.toList());
List<ProductInfoOutput> productInfoList=productClient.listForOrder(productIdList);// ***调用商品服务,获取商品列表***
//读redis
//减库存并将新值重新设置进redis
//订单入库异常,需要手动回滚redis
//计算总价
BigDecimal orderAmount=new BigDecimal(BigInteger.ZERO);
for(OrderDetail orderDetail:orderDTO.getOrderDetailList()){
for(ProductInfoOutput productInfo:productInfoList){
if(productInfo.getProductId().equals(orderDetail.getProductId())){
orderAmount=productInfo.getProductPrice().multiply(new BigDecimal(orderDetail.getProductQuantity())).add(orderAmount);
BeanUtils.copyProperties(productInfo,orderDetail);
orderDetail.setOrderId(orderId);
orderDetail.setDetailId(KeyUtil.genUniqueKey());
//订单详情入库
orderDetailRepository.save(orderDetail);
}
}
}
//扣库存(调用商品服务)
List<DecreaseStockInput> decreaseStockInputList=orderDTO.getOrderDetailList().stream()
.map(e ->new DecreaseStockInput(e.getProductId(),e.getProductQuantity())).collect(Collectors.toList());
productClient.decreaseStock(decreaseStockInputList);//***调用商品服务。减库存***
//订单入库
OrderMaster orderMaster=new OrderMaster();
BeanUtils.copyProperties(orderDTO,orderMaster);
orderMaster.setOrderAmount(orderAmount);
orderMaster.setOrderStatus(OrderStatusEnum.NEW.getCode());
orderMaster.setPayStatus(PayStatusEnum.WAIT.getCode());
orderMasterRepository.save(orderMaster);
return orderDTO;
}
}