章
目
录
一般情况下,服务提供者也不会是单机版,应该也是个集群(参考之前的Eureka原理图),下面我们先来搭建下服务提供者的集群,然后再做负载均衡调用。
第1步:创建Module
我们参考之前的cloud-provider-payment8001
子模块的创建,再创建个名为cloud-provider-payment8002
的子模块。
第2步:改pom
复制8001
的pom.xml
中的依赖,我们该子模块的pom.xml
如下:
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <parent> <artifactId>mscloud</artifactId> <groupId>com.panziye.springcloud</groupId> <version>1.0-SNAPSHOT</version> </parent> <modelVersion>4.0.0</modelVersion> <artifactId>cloud-provider-payment8002</artifactId> <dependencies> <!-- eureka client --> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependency> <!-- 引入自定义的api通用包,可以使用支付Payment等Entity --> <dependency> <groupId>com.panziye.springcloud</groupId> <artifactId>cloud-api-commons</artifactId> <version>${project.version}</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency> <dependency> <groupId>org.mybatis.spring.boot</groupId> <artifactId>mybatis-spring-boot-starter</artifactId> </dependency> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid-spring-boot-starter</artifactId> <version>1.2.5</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-devtools</artifactId> <scope>runtime</scope> <optional>true</optional> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <optional>true</optional> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> </dependencies> </project>
第3步:写yml
在resources
下新建application.yml
,复制8001
的yml,稍作修改,如下:
server: port: 8002 spring: application: name: cloud-payment-service datasource: type: com.alibaba.druid.pool.DruidDataSource driver-class-name: com.mysql.cj.jdbc.Driver url: jdbc:mysql://localhost:3306/dbcloud?serverTimezone=UTC&characterEncoding=utf8&useUnicode=true&useSSL=false username: root password: 123456 mybatis: mapperLocations: classpath:mapper/*.xml type-aliases-package: com.panziye.springcloud.entities eureka: client: #表示是否将自己注册进eureka服务中心,默认true register-with-eureka: true #表示是否从EurekaServer抓取已有注册信息,默认true。单节点无所谓,集群必须设置true才能配合ribbon使用负载均衡 fetch-registry: true service-url: #指向eureka集群 defaultZone: http://eureka7001.com:7001/eureka,http://eureka7002.com:7002/eureka,http://eureka7003.com:7003/eureka
第4步:主启动类
在com.panziye.springcloud
包下新建名为PaymentMain8002
主启动类,如下:
package com.panziye.springcloud; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cloud.netflix.eureka.EnableEurekaClient; @SpringBootApplication @EnableEurekaClient public class PaymentMain8002 { public static void main(String[] args) { SpringApplication.run(PaymentMain8002.class,args); } }
第5步:业务类
复制8001
的controller
、service
、dao
、mapper
文件到8002
项目对应目录下。
第6步:修改PaymentController
为了方便测试哪个模块提供了服务,我们修改下8001
和8002
中的PaymentController
,添加上端口的打印。
1)新增代码如下:
//springframework包下@Value获取application.yml中的配置 @Value("${server.port}") private String serverPort;
2)在查询操作成功是修改打印代码如下:
return new CommonResult(200,"查询数据库成功,serverPort: "+serverPort,payment);
第7步:测试
我们先启动3个注册中心(7001,7002,7003),再启动两个支付模块(8001,8002),最后启动订单模块(80),我们发现:
1)访问3个注册中心,发现支付模块有两个服务注册进去了(8001和8002)
2)我们多次访问get方法,始终调用的端口都是8001,表明始终调的是8001模块,原因是因为我们在cloud-consumer-order80
中的OrderController
将请求URL写死为http://localhost:8001
第8步:实现负载均衡
1)我们修改cloud-consumer-order80
中的OrderController
中的请求URL为服务在Eureka Server
的应用名称,即:CLOUD-PAYMENT-SERVICE
,之后我们服务调用只认应用名称,具体修改如下:
public static final String PAYMENT_URL = "http://CLOUD-PAYMENT-SERVICE";
2)重启后测试,再次访问,发现报错
3)解决:使用@LoadBalanced
注解赋予RestTemplate负载均衡的能力
我们在cloud-consumer-order80
中的ApplicationContextCofig
类中,添加注解,修改如下:
package com.panziye.springcloud.config; import org.springframework.cloud.client.loadbalancer.LoadBalanced; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.client.RestTemplate; @Configuration public class ApplicationContextCofig { /** * 配置RestTemplate * @return */ @Bean @LoadBalanced public RestTemplate getRestTemplate(){ return new RestTemplate(); } }