【SpringCloud】(二)Eureka注册中心和Feign远程调用
1 SpringCloud 核心
SpringCloud基于HTTP协议,这是和Dubbo最本质的区别,Dubbo的核心是RPC(远程方法调用)
Eureka:注册中心
Ribbon:客户端负载均衡
Feign:远程接口调用
Hystrix:服务的熔断、降级、监控
Zuul:网关

在客户端进行远程方法调用时,
Ribbon称为客户端的负载均衡,它决定Consumer向集群中的哪一个Provider发送请求。而
Feign是以Ribbon为基础的,Feign是一个声明式调用,客户端只需要通过装配调用Provider暴露出来的接口即可。
Hystrix则是提供了服务的熔断和降级,熔断是Provider端提供的Provider提供的服务无法响应时应急备用方案,降级是Consumer端提供的Provider提供的服务无法响应时应急备用方案。
HystrixDashBoard是Hystrix提供的监视整个服务状态的仪表盘。
Zuul网关作为服务的统一入口,zuul能够找到各种服务的原因也是以Ribbon为基础的,Ribbon通过查询Eureka注册中心找到各种服务。
2 SpringCloud测试
2.1 创建测试环境
① 新建Spring-cloud-parnet作为父工程,设置pom文件中打包方式为pom并在dependencyManagement中导入SpringCloud以及SpringBoot需要的依赖。
<?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">
<modelVersion>4.0.0</modelVersion>
<groupId>org.example</groupId>
<artifactId>spring-cloud-parnet</artifactId>
<packaging>pom</packaging>
<version>1.0-SNAPSHOT</version>
<modules>
<module>spring-cloud-common</module>
<module>spring-cloud-provider</module>
<module>spring-boot-consumer</module>
</modules>
<properties>
<maven.compiler.source>8</maven.compiler.source>
<maven.compiler.target>8</maven.compiler.target>
</properties>
<dependencies>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>
</dependencies>
<dependencyManagement>
<dependencies>
<dependency>
<!-- SpringCloud需要的依赖 -->
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>Hoxton.SR9</version>
<type>pom</type>
<!-- 将 spring-cloud-dependencies 中的依赖全部导入 -->
<scope>import</scope>
</dependency>
<dependency>
<!-- SpringBoot需要的依赖 -->
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.3.7.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
</project>
scope import用来导入Spring-boot-denpencies中的所有依赖,是为了实现maven的多继承,可以参考我之前的maven笔记:博客园
② 创建Spring-cloud-common maven模块,用于存放Employee实体
public class Employee {
private int empId;
private String empName;
private double empSalary;
public Employee() {
}
public Employee(int empId, String empName, double empSalary) {
this.empId = empId;
this.empName = empName;
this.empSalary = empSalary;
}
public int getEmpId() {
return empId;
}
public void setEmpId(int empId) {
this.empId = empId;
}
public String getEmpName() {
return empName;
}
public void setEmpName(String empName) {
this.empName = empName;
}
public double getEmpSalary() {
return empSalary;
}
public void setEmpSalary(double empSalary) {
this.empSalary = empSalary;
}
}
在父工程下创建的模块会自动继承并导入父工程的依赖,以及在子工程引入的依赖需要收到父工程中dependencyManagement依赖管理器中设置的相关依赖版本的限制。
③ 创建生产者和消费者模块,在pom文件中导入spring-boot-starter-web以及common模块(懒就不写了)
设置生产者的端口号为1000,消费者访问端口号为4000,生产者控制器为:
@RestController
public class EmployeeController {
@RequestMapping("/provider/get/employee/remote")
public Employee getEmployeeRemote() {
return new Employee(5, "5555", 5555);
}
}
2.2 RestTemplate远程调用测试
⑤ 在消费者控制器通过restTemplate访问远程端口
首先,将RestTemplate注入IOC容器
@Configuration
public class HikaruSpringCloudConfig {
@Bean
public RestTemplate restTemplate() {
return new RestTemplate();
}
}
然后,在消费者控制器中通过RestTemplate调用远程服务。
@RestController
public class HumanResourceController {
@Autowired
private RestTemplate restTemplate;
@RequestMapping("/consumer/get/employee")
public Employee getEmployeeRemote() {
String host = "http://localhost:1000";
String url = "/provider/get/employee/remote";
Employee employee = restTemplate.getForObject(host + url, Employee.class);
// 通过RestTemplate调用远程微服务
return employee;
}
@RequestMapping("/get/employee")
public Employee getEmployee() {
return new Employee(1, "11", 1111);
}
}
记录一个坑点
这里如果实体使用lombok的话远程调用是失败的,爆出的错误是json不支持没有构造器的对象转换,可以看到我在下面还写了一个直接返回的是能够得到的,原因不明。。
3 Eureka注册中心
3.1 引入依赖
创建spring-cloud-eureka并导入依赖:
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
</dependencies>
3.2 在启动类中开启Eureka服务
@EnableEurekaServer
@SpringBootApplication
public class HikaruMainType {
public static void main(String[] args) {
SpringApplication.run(HikaruMainType.class);
}
}
这里对SpringCloud的版本有要求的,原来使用的 Hoxton.SR9 版本的SpringCloud这里会出现@EnableEurekaServer注解找不到的情况
所以如下修改了父工程的依赖管理器中的SpringCloud版本号,SpringBoot的版本查询完官网后不需要改变
<dependencyManagement>
<dependencies>
<dependency>
<!-- SpringCloud需要的依赖 -->
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>Hoxton.SR12</version>
<type>pom</type>
<!-- 将 spring-cloud-dependencies 中的依赖全部导入 -->
<scope>import</scope>
</dependency>
<dependency>
<!-- SpringBoot需要的依赖 -->
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.3.7.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
3.3 配置Eeureka
server:
port: 5000
eureka:
instance:
hostname: localhost # 配置当前Eureka服务的主机地址
client:
register-with-eureka: false # 是否需要进行eureka注册
fetch-registry: false # 是否从eureka中取回信息
service-url: # 客户端(指的是consumer、provider)访问当前注册中心时使用的地址
defaultZone: http://${eureka.instance.hostname}/${server.port}/eureka
eureka自身自然不需要注册和获取信息了
3.4 启动Eureka(一些小坑
这里启动搞了很久,最后修改SpringCloud和SpringBoot版本分别为Hoxton.SR12和2.3.12.RELEASE:
<dependencyManagement>
<dependencies>
<dependency>
<!-- SpringCloud需要的依赖 -->
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-dependencies</artifactId>
<version>Hoxton.SR12</version>
<type>pom</type>
<!-- 将 spring-cloud-dependencies 中的依赖全部导入 -->
<scope>import</scope>
</dependency>
<dependency>
<!-- SpringBoot需要的依赖 -->
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.3.12.RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
然后启动报找不到AbstractDiscoveryClientOptionalArgs Bean的错误需要导入spring-boot-starter-web
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>

然后在网址输入http://localhost:5000/显示上面页面即成功,# Instances currently registered with Eureka表示注册到Eureka的实例。
3.5 注册Provider
① 在provider中导入依赖
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
② 配置eureka
server:
port: 1000
eureka:
client:
service-url:
defaultZone: http://localhost:5000/eureka
spring:
application:
name: hikaru-provider
配置完成之后即可在页面看到注册的Instances,名称即为设置的name
3.6 调整consumer:使用微服务名称进行远程调用

分析:consumer通过配置的eureka客户端的ribbon按照微服务名查找eureka注册中心中的实例,然后由eureka将微服务名称转换为真实地址(ip+端口号),由restTemplate进行具体调用并将结果返回给consumer。restTemplate添加@LoadBanlanced可以实现客户端的负载均衡。
① 导入依赖
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
eureka底层自带ribbon所以不需要再添加ribbon了
② 配置eureka客户端并设置服务名称
server:
port: 4000
spring:
application:
name: hikaru-consumer
eureka:
client:
service-url:
defaultZone: http://localhost:5000/eureka
③ restTemplate开启负载均衡
@Configuration
public class HikaruSpringCloudConfig {
@Bean
@LoadBalanced
public RestTemplate restTemplate() {
return new RestTemplate();
}
}
@LoadBanlanced让RestTemplate有负载均衡的功能,能够访问provider的集群。
④ 调整controller测试
@RequestMapping("/consumer/eureka/get/employee")
public Employee getEmployeeRemote() {
//String host = "http://localhost:1000";
String host = "http://hikaru-provider";
String url = "/provider/get/employee/remote";
Employee employee = restTemplate.getForObject(host + url, Employee.class);
// 通过RestTemplate调用远程微服务
return employee;
}
如此便实现了使用服务名代替地址进行远程调用
@EnableDiscoveryClient 和 @EnableEurekaClient都是开启注册中心客户端服务,只不过前者不局限于eureka(如nacos),并且在较高版本的SpringBoot中都可以省略
3.7 以集群方式启动provider
idea在Run -》 edit configration中右上角设置Allow parallel run即可同时运行多个实例,然后分别运行eureka、consumer以及端口号为1000 2000 3000 的provider,这时候的eureka页面为:

然后多次访问 \http://localhost:4000/consumer/eureka/get/employee发现是以3000 2000 1000的轮调的顺序访问的
本质是ribbon拿着服务名去eureka中找到的三个端口号再去请求的服务。
4 Feign远程方法调用
4.1 Feign的使用
① 创建一个使用feign的consumer模块
② 在创建的模块和common模块中导入openfeign依赖
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
③ 创建接口 EmployeeRemoteService,并使用FeignClient注解绑定接口和一个微服务名称
@FeignClient("hikaru-provider")
public interface EmployeeRemoteService {
/**
* 远程调用的接口方法
* 要求:
* ① @RequestMapping地址一致
* ② 方法声明一致
* ③ 用来获取请求参数的@RequestParam、@PathVariable、@RequestBody一致
* @return
*/
@RequestMapping("/provider/get/employee/remote")
public Employee getEmployeeRemote();
}
远程调用的接口方法
要求:
① @RequestMapping地址一致
② 方法声明一致
③ 用来获取请求参数的@RequestParam、@PathVariable、@RequestBody一致
④ 对应的provider的远程方法
@RestController
public class EmployeeController {
@RequestMapping("/provider/get/employee/remote")
public Employee getEmployeeRemote() {
return new Employee(5, "5555", 5555);
}
⑤ 在feignConsumer中开启FeignClients
@EnableFeignClients
@SpringBootApplication
public class HikaruMainType {
public static void main(String[] args) {
SpringApplication.run(HikaruMainType.class);
}
}
⑥ 在feinConsumer装配使用common模块的接口方法
@RestController
public class FeignHumanResource {
@Autowired
EmployeeRemoteService employeeRemoteService;
@RequestMapping("/feign/consumer/get/emp")
public Employee getEmployee() {
return employeeRemoteService.getEmployeeRemote();
}
}
这里装配一个接口能够成功就是因为接口类通过 @FeignClient("hikaru-provider")绑定了一个微服务应用,然后feign会根据方法再在底层由
retemplate调用ribbon去eureka注册中心去找对应微服务应用名的实例,调用实例暴露的方法然后返回。

如上图,其实是为了解耦将一个微服务Provider提供的接口服务抽取出来放到了common模块,这样就不需要在每次远程调用都去使用@FeignClient("hikaru-provider")去绑定微服务应用。
使用Feign实现了对resTemplate的封装,从而也不需要创建配置类配置resTemplate;而在consumer中只需要使用注解@EnableFeignClients开启feign客户端,其他实现和单一架构完全相同;而远程服务结构的绑定则是在写在common模块使用注解@FeignClient("hikaru-provider")实现
【SpringCloud】(二)Eureka注册中心和Feign远程调用的更多相关文章
- SpringCloud(二) - Eureka注册中心,feign远程调用,hystrix降级和熔断
1.项目模块介绍 2. 父项目 主要依赖 spring-cloud 的 版本控制 <properties> <!-- springCloud 版本 --> <scd.ve ...
- springcloud(二):注册中心Eureka
Eureka是Netflix开源的一款提供服务注册和发现的产品,它提供了完整的Service Registry和Service Discovery实现.也是springcloud体系中最重要最核心的组 ...
- SpringCloud之Eureka注册中心原理及其搭建
一.Eureka简介 Eureka是Netflix开发的服务发现框架,本身是一个基于REST的服务,主要用于定位运行在AWS域中的中间层服务,以达到负载均衡和中间层服务故障转移的目的.SpringCl ...
- SpringCloud之eureka注册中心入门
eureka注册中心 一.基本概念 SpringCloud封装 了Netflix公司的eureka作为自己微服务的注册中心.这个注册中心和dubbo中的zookeeper很相似,简单来说,只要你可以将 ...
- SpringCloud(一)Eureka注册中心
Eureka简介 Eureka作为注册中心,管理各种服务功能包括服务的注册.发现.熔断.负载.降级等 Eureka注册中心实例 Eureka Server 1.pom文件配置SpringBoot.Sp ...
- F版本SpringCloud 4—Eureka注册中心开发和客户端开发
源码地址:https://gitee.com/bingqilinpeishenme/Java-Tutorials 前言 通过前三篇文章,用大白话介绍了微服务和SpringCloud以及服务治理相关的概 ...
- springcloud之Eureka注册中心
参考博客:https://www.cnblogs.com/ityouknow/p/6854805.html 背景: Eureka是Netflix开源的一款提供服务注册和发现的产品,它提供了完整的Ser ...
- Spring-Cloud之Eureka注册中心环境搭建(单节点)
一 Eureka概述 服务启动时会生成服务的基本信息对象InstanceInfo,然后在启动时会register到服务治理中心. 注册完成后会从服务治理中心拉取所有的服务信息,缓存在本地. 之后服务会 ...
- SpringCloud(二)Eureka集群与Feign
两个Eureka 上一篇是两个服务像一个Eureka注册,如果Eureka宕掉了就不好了,现在来搭建两个Eureka,两个服务分别像其注册.两个Eureka都用本机来模拟,用两个端口号来表示. 首先修 ...
- springCloud 之 Eureka注册中心高可用配置
springCloud的eureka高可用配置方案思路是:几个服务中心之间相互注册,比如两个注册中心,A注册到B上,B注册到A上,如果是三个注册中心则是:A注册到BC上,B注册到AC上,C注册到AB上 ...
随机推荐
- mmdetection可视化工具-DetVisGUI
保存数据 执行程序,需要保存输出结果的pkl文件或者json文件 下面以测试faster_rcnn示例: 在执行测试时可以使用下面这条命令,就会将结果保存到一个pkl文件中. python tools ...
- vue使用echarts控制台报错Can't get DOM width or height并且地图显示超范围
用echarts实现展示地图,但是地图显示的范围一直超出他那个div,同时报错. 完整报错信息: Can't get DOM width or height. Please check dom.cli ...
- MySQL -my.cnf配置文件优化
# [mysqld] datadir=/var/lib/mysql #socket=/var/lib/mysql/mysql.sock user=mysql ### 设置主从的时候的唯一ID 每台主机 ...
- 制作mnist格式数据集
import os from PIL import Image from array import * from random import shuffle # # 文件组织架构: # ├──trai ...
- weblogic session timed out
How to Configure Session Timeout in Weblogic Server (WLS) ?
- Leecode 160.相交链表(Java 哈希表、双指针 两种方法)
找两个链表第一次指针相同的地方 想法:(本来是没有的,因为没读懂题目描述= =) 1.两个指针,长的先走(长减短相差的长度)这么多的步数,然后就可以开始比较指针,直到指向为空,期间如果指针相同 ...
- idea开发使用外置tomcat配置
1. 添加依赖 <dependency> <groupId>org.apache.tomcat.embed</groupId> <artifactId> ...
- 智汇成城 ,创赢未来 | AI+产业峰会智慧城市专场在深成功举办!
11月4日下午,由福田区人才工作局指导,广州英码信息科技有限公司和共达地创新技术(深圳)有限公司联合主办,深圳市人工智能行业协会承办的AI+产业峰会之智慧城市专场活动在深圳市南山区成功举办. &quo ...
- 飞桨AI 文本实体抽取 数据准备(excel 文本标注)
网纸: https://ai.baidu.com/easydl/app/deploy/tee/public #!/usr/bin/env python3 # -*- coding: utf-8 -*- ...
- celery 使用
celery 1.celery介绍 celery能用来做什么: 1.异步任务 2.定时任务 3.延迟任务 1.1 理解celery的运行原理 1.可以不依赖任何服务器 通过自身命令 启动服务 2.ce ...