【微服务学习二】nacos服务发现与负载均衡

nacos服务发现

想要开启服务发现,需要在main函数上添加 @EnableDiscoveryClient 注解

然后我们编写一个controller类来查询nacos中注册的所有微服务以及对应的ip+端口号

java 复制代码
@Controller
public class DiscoveryController {

    @Autowired
    DiscoveryClient discoveryClient;

    // 这个方法可以获取nacos中注册的服务信息
    @GetMapping("/getServicesInfo")
    @ResponseBody
    public String getServicesInfo() {
        StringBuilder sb = new StringBuilder();
        // 查询服务名称和服务的ip+端口号
        for (String serviceId : discoveryClient.getServices()) {
            sb.append(serviceId + "<br>");
            for (ServiceInstance instance : discoveryClient.getInstances(serviceId)) {
                sb.append(instance.getHost() + ":" + instance.getPort() + "<br>");
            }
        }
        return sb.toString();
    }

}

接着启动微服务,然后在浏览器中调用该接口就可以查到对应的信息

简单下单场景

了解了服务发现的基本内容,我们就可以进行一个简单的下单场景模拟,场景流程如下

首先我们需要给商品服务编写一个根据商品id查询商品的接口

由于我们需要编写bean类,为了方便起见,我们首先给services的配置文件中添加lombok依赖和fastjson依赖

接着我们编写controller类,这个类提供一个查询商品信息的接口

java 复制代码
@RestController
public class ProductController {

    @Autowired
    ProductService productService;

    @GetMapping("/getProduct/{id}")
    @ResponseBody
    public String getProductById(@PathVariable("id") Long productId) {
        Product product = productService.getProductById(productId);
        return JSON.toJSONString(product);
    }

}

接着是商品的bean类

java 复制代码
@Data
public class Product {

    private Long id;
    private BigDecimal price;
    private String productName;
    private int num;

}

最后是service类

java 复制代码
@Service
public class ProductServiceImpl implements ProductService {

    @Override
    public Product getProductById(Long id) {
        Product product = new Product();
        product.setId(id);
        product.setPrice(new BigDecimal("99"));
        product.setProductName("苹果-" + id);
        product.setNum(2);
        return product;
    }
}

在浏览器中调用方法可以获取查询结果

由于微服务之间可能会有相互依赖的模型层,因此需要将共用的模型层的bean类放到一个公共的模块中

需要在model模块中添加lombok依赖

然后在services模块中引入model模块

一下是Order实体类的代码

java 复制代码
@Data
public class Order {

    private Long id;
    private BigDecimal totalAmount;
    private Long userId;
    private String nickName;
    private String address;
    private List<Object> productList;

}

接着我们编写order服务的controller

java 复制代码
@RestController
public class OrderController {

    @Autowired
    OrderService orderService;

    @GetMapping("/createOrder/{userId}/{productId}")
    public String createOrder(@PathVariable Long userId, @PathVariable Long productId) {
        Order order = orderService.createOrder(userId, productId);
        return JSON.toJSONString(order);
    }

}

由于在service中由于需要使用到RestTemplate进行远程调用,因此我们需要编写一个配置类,给容器注入一个RestTemplate实例

java 复制代码
@Configuration
public class OrderConfig {

    @Bean
    public RestTemplate getRestTemplate() {
        return new RestTemplate();
    }

}

最后编写service

java 复制代码
@Slf4j
@Service
public class OrderServiceImpl implements OrderService {

    @Autowired
    DiscoveryClient discoveryClient;

    @Autowired
    RestTemplate restTemplate;

    @Override
    public Order createOrder(Long userId, Long productId) {
        Order order = new Order();
        order.setId(2L);
        order.setUserId(userId);
        Product product = getProductRemote(productId);
        BigDecimal totalPrice = product.getPrice().multiply(new BigDecimal(product.getNum()));
        order.setTotalAmount(totalPrice);
        order.setAddress("福建省");
        order.setNickName("小明明");
        order.setProductList(Arrays.asList(product));
        return order;
    }

    // 通过远程调用的方式获取商品数据
    private Product getProductRemote(Long productId) {
        discoveryClient.getInstances("service-product");
        List<ServiceInstance> instances = discoveryClient.getInstances("service-product");
        String url = "http://" + instances.get(0).getHost() + ":" + instances.get(0).getPort() + "/getProduct/" + productId;
        log.info("远程调用:{}", url);
        Product product = restTemplate.getForObject(url, Product.class);
        return product;
    }


}

在浏览器中调用接口就可以获取数据

负载均衡

由于我们的订单服务需要进行负载均衡的调用,因此首先需要添加负载均衡的依赖

接着就是在service中使用 LoadBalancerClient 进行负载均衡的调用

用浏览器进行多次调用得到的测试结果如下

可以发现是交替调用两个商品服务的

除了上述方式外,还可以通过注解的方式进行负载均衡的调用

首先需要在配置类中加上 @LoadBalanced 注解

接着在如下图修改远程调用方法中的url即可

相关推荐
Chef_Chen1 小时前
从0开始学习计算机视觉--Day08--卷积神经网络
学习·计算机视觉·cnn
weixin_446260851 小时前
Isaac Lab:让机器人学习更简单的开源框架
学习·机器人
我真不会起名字啊3 小时前
OpenSceneGraph(OSG)开发学习
学习
永日456703 小时前
学习日记-spring-day42-7.7
java·学习·spring
Love__Tay6 小时前
笔记/云计算基础
笔记·学习·云计算
wuxuanok9 小时前
Web后端开发-分层解耦
java·笔记·后端·学习
wuxuanok9 小时前
Web后端开发-请求响应
java·开发语言·笔记·学习
i7i8i9com10 小时前
后端微服务基础架构Spring Cloud
学习
蜡笔小电芯10 小时前
【C语言】指针与回调机制学习笔记
c语言·笔记·学习
Code季风10 小时前
Gin Web 层集成 Viper 配置文件和 Zap 日志文件指南(下)
前端·微服务·架构·go·gin