【转】 Spring Boot + Eureka 实现微服务负载均衡

1,什么是Eureka,什么是服务注册与发现

      Spring Boot作为目前最火爆的web框架。那么它与Eureka又有什么关联呢?

    Eureka是Netflix开源的一个RESTful服务,主要用于服务的注册发现。

    Eureka由两个组件组成:Eureka服务器和Eureka客户端。Eureka服务器用作服务注册服务器。

    Eureka客户端是一个java客户端,用来简化与服务器的交互、作为轮询负载均衡器,并提供服务的故障切换支持。

    Netflix在其生产环境中使用的是另外的客户端,它提供基于流量、资源利用率以及出错状态的加权负载均衡。

2,先创建一个Eureka-Server服务注册中心


    这里需要用到spring-cloud的Eureka模块,他是一个服务的注册和发现模块

      如图我们先new一个Spring-boot工程引入Eureka Server


Next>>>>Finish完成

我们来看看构建好的Eureka-Server的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">

        <modelVersion>4.0.0</modelVersion>

        <groupId>com.eureka</groupId>

        <artifactId>server</artifactId>

        <version>0.0.1-SNAPSHOT</version>

        <packaging>jar</packaging>

        <name>server</name>

        <description>Demo project for Spring Boot</description>

        <parent>

            <groupId>org.springframework.boot</groupId>

            <artifactId>spring-boot-starter-parent</artifactId>

            <version>2.0.2.RELEASE</version>

            <relativePath/> <!-- lookup parent from repository -->

        </parent>

        <properties>

            <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>

            <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>

            <java.version>1.8</java.version>

            <spring-cloud.version>Finchley.RC2</spring-cloud.version>

        </properties>

        <dependencies>

            <!-- 引入的Eureka-server -->

            <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-test</artifactId>

                <scope>test</scope>

            </dependency>

            <dependency>

                <groupId>org.springframework.boot</groupId>

                <artifactId>spring-boot-autoconfigure</artifactId>

            </dependency>

        </dependencies>

        <dependencyManagement>

            <dependencies>

                <dependency>

                    <groupId>org.springframework.cloud</groupId>

                    <artifactId>spring-cloud-dependencies</artifactId>

                    <version>${spring-cloud.version}</version>

                    <type>pom</type>

                    <scope>import</scope>

                </dependency>

            </dependencies>

        </dependencyManagement>

        <build>

            <plugins>

                <plugin>

                    <groupId>org.springframework.boot</groupId>

                    <artifactId>spring-boot-maven-plugin</artifactId>

                </plugin>

            </plugins>

        </build>

        <repositories>

            <repository>

                <id>spring-milestones</id>

                <name>Spring Milestones</name>

                <url>https://repo.spring.io/milestone</url>

                <snapshots>

                    <enabled>false</enabled>

                </snapshots>

            </repository>

        </repositories>

    </project>

    我们看到这里与普通的Spring-boot项目不同的是,这里引用了一个Eureka-Server包。

    那么我们怎么使用它呢,怎么启动它呢?


    这里只需要启动一个注解就可以啦,我们在Spring-Boot工程的启动类上加>>>>>>

    @EnableEurekaServer

    代码如下:


    package com.eureka.server;

    import org.springframework.boot.SpringApplication;

    import org.springframework.boot.autoconfigure.SpringBootApplication;

    import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer;

    /**

    * 启动一个服务注册中心

    */

    @EnableEurekaServer

    @SpringBootApplication

    public class ServerApplication {


        public static void main(String[] args) {

            SpringApplication.run(ServerApplication.class, args);

        }

    }

差点忘了,我们还需要配置application.yml

Eureka是一个高可用的组件,每一个实例注册之后需要向注册中心发送心跳包,在默认情况下erureka server也是一个eureka client ,必须要指定一个 server。

eureka server的配置文件appication.yml:

    server:

      port: 8081 #服务注册中心端口号

    eureka:

      instance:

        hostname: 127.0.0.1 #服务注册中心IP地址

      client:

        registerWithEureka: false #是否向服务注册中心注册自己

        fetchRegistry: false #是否检索服务

        serviceUrl: #服务注册中心的配置内容,指定服务注册中心的位置

          defaultZone: http://${eureka.instance.hostname}:${server.port}/eureka/

我们来启动一下吧

我们在浏览器上输入http://127.0.0.1:8081/飞机直达

我们可以看到它的可视化界面

细心的朋友会发现,这里没有发现服务???No instance available

why?  因为我们还没有服务向注册中心注册服务,所以找不到啊。

3,先创建一个Eureka-Client客户端也就是服务提供者

        客户端在向注册中心它会提供一些元数据,例如主机和端口,URL,主页等。Eureka server 从    每        个client实例接收心跳消息。 如果心跳超时,则通常将该实例从注册server中删除。     

    创建客户端和服务端差不多,只是启动注解有点不一样,还有yml配置文件


Next>>>Finish完成啦

打开会发现pom.xml其实和Server注册中心的类似

    <?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>com.eureka</groupId>

        <artifactId>provider</artifactId>

        <version>0.0.1-SNAPSHOT</version>

        <packaging>jar</packaging>

        <name>provider</name>

        <description>Demo project for Spring Boot</description>

        <parent>

            <groupId>org.springframework.boot</groupId>

            <artifactId>spring-boot-starter-parent</artifactId>

            <version>2.0.2.RELEASE</version>

            <relativePath/> <!-- lookup parent from repository -->

        </parent>

        <properties>

            <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>

            <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>

            <java.version>1.8</java.version>

            <spring-cloud.version>Finchley.RC2</spring-cloud.version>

        </properties>

        <dependencies>

            <dependency>

                <groupId>org.springframework.boot</groupId>

                <artifactId>spring-boot-starter-web</artifactId>

            </dependency>

            <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-test</artifactId>

                <scope>test</scope>

            </dependency>

        </dependencies>

        <dependencyManagement>

            <dependencies>

                <dependency>

                    <groupId>org.springframework.cloud</groupId>

                    <artifactId>spring-cloud-dependencies</artifactId>

                    <version>${spring-cloud.version}</version>

                    <type>pom</type>

                    <scope>import</scope>

                </dependency>

            </dependencies>

        </dependencyManagement>

        <build>

            <plugins>

                <plugin>

                    <groupId>org.springframework.boot</groupId>

                    <artifactId>spring-boot-maven-plugin</artifactId>

                </plugin>

            </plugins>

        </build>

        <repositories>

            <repository>

                <id>spring-milestones</id>

                <name>Spring Milestones</name>

                <url>https://repo.spring.io/milestone</url>

                <snapshots>

                    <enabled>false</enabled>

                </snapshots>

            </repository>

        </repositories>

    </project>

怎么证明它是Client呢

很简单

在Spring-boot的启动类上通过注解@EnableEurekaClient 表明自己是一个eurekaclient.

    package com.eureka.provider;

    import org.springframework.boot.SpringApplication;

    import org.springframework.boot.autoconfigure.SpringBootApplication;

    import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

    import org.springframework.web.bind.annotation.GetMapping;

    import org.springframework.web.bind.annotation.RequestParam;

    import org.springframework.web.bind.annotation.ResponseBody;

    import org.springframework.web.bind.annotation.RestController;

    import java.util.HashMap;

    import java.util.Map;

    /**

    * Eureka客户端

    */

    @RestController

    @EnableEurekaClient

    @SpringBootApplication

    public class ProviderApplication {


        public static void main(String[] args) {

            SpringApplication.run(ProviderApplication.class, args);

        }



        /**

        * 假如这个客户端要提供一个getUser的方法

        * @return

        */

        @GetMapping(value = "/getUser")

        @ResponseBody

        public Map<String,Object> getUser(@RequestParam Integer id){

            Map<String,Object> data = new HashMap<>();

            data.put("id",id);

            data.put("userName","admin");

            data.put("from","provider-A");

            return data;

        }


    }

虽然加好了@EnableEurekaClient,总感觉差点什么,对了,配置文件yml

eureka:

  client:

    serviceUrl: #注册中心的注册地址

      defaultZone: http://127.0.0.1:8081/eureka/

server:

  port: 8082  #服务端口号

spring:

  application:

    name: service-provider #服务名称--调用的时候根据名称来调用该服务的方法

我们来启动看看吧

我们看到这个客户端已经向注册中心注册服务了,那么我们打开Eureka-server飞机直达

我们看到我们启动的服务是不是加进去了呢

我们看到我们的服务是不是加进去了呢。

那么有人会问,那一大堆飙红的什么意思啊。因为注册的服务都是高可用的,这里只检测到一个服务,产生的预警,不影响使用,等下我们启动多个实例就不会了。

我们先来测试下客户端的方法是否可用  飞机直达

显然是没有问题,那么我们提供好了服务,sei来消费呢?

下面我们就来建立一个消费者

为了更简单易懂,我还是一步一步出图吧。

来,贴上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">

    <modelVersion>4.0.0</modelVersion>

    <groupId>com.eureka</groupId>

    <artifactId>consumer</artifactId>

    <version>0.0.1-SNAPSHOT</version>

    <packaging>jar</packaging>

    <name>consumer</name>

    <description>Demo project for Spring Boot</description>

    <parent>

        <groupId>org.springframework.boot</groupId>

        <artifactId>spring-boot-starter-parent</artifactId>

        <version>2.0.2.RELEASE</version>

        <relativePath/> <!-- lookup parent from repository -->

    </parent>

    <properties>

        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>

        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>

        <java.version>1.8</java.version>

        <spring-cloud.version>Finchley.RC2</spring-cloud.version>

    </properties>

    <dependencies>

        <dependency>

            <groupId>org.springframework.boot</groupId>

            <artifactId>spring-boot-starter-web</artifactId>

        </dependency>

        <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-test</artifactId>

            <scope>test</scope>

        </dependency>

    </dependencies>

    <dependencyManagement>

        <dependencies>

            <dependency>

                <groupId>org.springframework.cloud</groupId>

                <artifactId>spring-cloud-dependencies</artifactId>

                <version>${spring-cloud.version}</version>

                <type>pom</type>

                <scope>import</scope>

            </dependency>

        </dependencies>

    </dependencyManagement>

    <build>

        <plugins>

            <plugin>

                <groupId>org.springframework.boot</groupId>

                <artifactId>spring-boot-maven-plugin</artifactId>

            </plugin>

        </plugins>

    </build>

    <repositories>

        <repository>

            <id>spring-milestones</id>

            <name>Spring Milestones</name>

            <url>https://repo.spring.io/milestone</url>

            <snapshots>

                <enabled>false</enabled>

            </snapshots>

        </repository>

    </repositories>

</project>

主要是启动类,里面内容就丰富啦,都在注释里

package com.eureka.consumer;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.boot.SpringApplication;

import org.springframework.boot.autoconfigure.SpringBootApplication;

import org.springframework.cloud.client.loadbalancer.LoadBalanced;

import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

import org.springframework.context.annotation.Bean;

import org.springframework.web.bind.annotation.GetMapping;

import org.springframework.web.bind.annotation.RequestParam;

import org.springframework.web.bind.annotation.ResponseBody;

import org.springframework.web.bind.annotation.RestController;

import org.springframework.web.client.RestTemplate;

import java.util.HashMap;

import java.util.Map;

/**

* Eureka客户端-消费者

*/

@RestController

@EnableEurekaClient

@SpringBootApplication

public class ConsumerApplication {

    @Autowired

    RestTemplate restTemplate;

    public static void main(String[] args) {

        SpringApplication.run(ConsumerApplication.class, args);

    }

    /**

    * 实例化RestTemplate

    * @return

    */

    @LoadBalanced

    @Bean

    public RestTemplate rest() {

        return new RestTemplate();

    }

    /**

    * Rest服务端使用RestTemplate发起http请求,然后得到数据返回给前端----gotoUser是为了区分getUser怕小伙伴晕头

    * @param id

    * @return

    */

    @GetMapping(value = "/gotoUser")

    @ResponseBody

    public Map<String,Object> getUser(@RequestParam Integer id){

        Map<String,Object> data = new HashMap<>();

        /**

        * 小伙伴发现没有,地址居然是http://service-provider

        * 居然不是http://127.0.0.1:8082/

        * 因为他向注册中心注册了服务,服务名称service-provider,我们访问service-provider即可

        */

        data = restTemplate.getForObject("http://service-provider/getUser?id="+id,Map.class);

        return data;

    }

}

配置文件和

eureka:

  client:

    serviceUrl: #注册中心的注册地址

      defaultZone: http://127.0.0.1:8081/eureka/

server:

  port: 8083  #服务端口号

spring:

  application:

    name: service-consumer #服务名称--调用的时候根据名称来调用该服务的方法

我们启动看看效果吧

看看我们的提供者和消费者是不是都进来了

那么我们看看我们消费者的方法是否可用  飞机直达

哈哈,是不是很神奇

下面介绍个更神奇的东西--实现微服务负载均衡

我们把服务提供者复制一个工程出来,我们再做下小小的修改,看看是否能实现负载均衡。

我们需要修改两个文件

一个是启动类,改了哪些呢?看看就晓得咯

package com.eureka.provider;

import org.springframework.boot.SpringApplication;

import org.springframework.boot.autoconfigure.SpringBootApplication;

import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

import org.springframework.web.bind.annotation.GetMapping;

import org.springframework.web.bind.annotation.RequestParam;

import org.springframework.web.bind.annotation.ResponseBody;

import org.springframework.web.bind.annotation.RestController;

import java.util.HashMap;

import java.util.Map;

/**

* Eureka客户端

*/

@RestController

@EnableEurekaClient

@SpringBootApplication

public class ProviderApplication {

    public static void main(String[] args) {

        SpringApplication.run(ProviderApplication.class, args);

    }

    /**

    * 假如这个客户端要提供一个getUser的方法

    * @return

    */

    @GetMapping(value = "/getUser")

    @ResponseBody

    public Map<String,Object> getUser(@RequestParam Integer id){

        Map<String,Object> data = new HashMap<>();

        data.put("id",id);

        data.put("userName","admin");

        data.put("from","provider-B");//改这里是为了让大家更能理解它负载均衡的机制

        return data;

    }

}

还有就是yml配置文件

eureka:

  client:

    serviceUrl: #注册中心的注册地址

      defaultZone: http://127.0.0.1:8081/eureka/

server:

  port: 8088  #服务端口号--该端口不要冲突

spring:

  application:

    name: service-provider #服务名称--调用的时候根据名称来调用该服务的方法--名字绝对不能改,改了就访问不到了

我们来启动一下吧

看看Eureka-server后台的效果    ServerA      ServerB   

这个叫做Service-provider是不是有两个实例啊

那么,我们分别访问一下,看看效果怎么样

看到了吗,8082端口,from是provider-A,8088端口,from是provider-B.

那么我们访问消费者的服务器看看会出现什么样的情况呢  飞机直达

一开始是from A,你刷新一下,诶? 变成 from B了。

说明这个时候两台提供者在交替工作,从而达到了一个负载均衡的作用。

来来来,我给你画个图

每个微服务都是一个Eureka-Client,我们把每个app(SpringBootApplication)都向注册中心注册一个服务。

有时候,某个服务的工作量比较大的时候,我们可以多注册几个同名称的微服务,从而让他们交替工作,减轻单个服务的压力。

————————————————

版权声明:本文为CSDN博主「TIny1995」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/nanbiebao6522/article/details/80574463

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容