美文网首页我爱编程
Spring cloud 简单实现

Spring cloud 简单实现

作者: 田大娃的辣条 | 来源:发表于2018-02-05 15:34 被阅读0次

    spring cloud

    在微服务架构中,业务都会被拆分成一个独立的服务,服务与服务的通讯是基于http restful的。

    Spring提供了一系列工具,可以帮助开发人员迅速搭建分布式系统中的公共组件,协调分布式环境中各个系统,为各类服务提供模板性配置

    比如:配置管理,服务发现,断路器,智能路由,微代理,控制总线,一次性令牌,全局锁,主节点选举, 分布式session, 集群状态等等。

    另外说明spring cloud是基于springboot的

    既然是管理分布式环境下的微服务,必然存在服务的注册问题。所以我们先从服务的注册谈起。既然是注册,必然有个管理注册中心的服务器,各个在Spring Cloud管理下的Spring Boot应用就是需要注册的client

    Eureka  是一个服务注册和发现模块。

    启动一个服务注册中心,只需要一个注解@EnableEurekaServer

    eureka是一个高可用的组件,它没有后端缓存,每一个实例注册之后需要向注册中心发送心跳(因此可以在内存中完成),在默认情况下erureka server也是一个eureka client ,必须要指定一个 server

    通过eureka.client.registerWithEureka:false和fetchRegistry:false来表明自己是一个eureka server.

    server:

      port: 8761

    eureka:

      instance:

        hostname: localhost

      client:

        registerWithEureka: false

        fetchRegistry: false

        serviceUrl:

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

    服务提供者 (eureka client)

    当client向server注册时,它会提供一些元数据,例如主机和端口,URL,主页等。Eureka server 从每个client实例接收心跳消息。

    如果心跳超时,则通常将该实例从注册server中删除。

    通过注解@EnableEurekaClient 表明自己是一个eurekaclient.

    需要在配置文件中注明自己的服务注册中心的地址spring.application.name,这个很重要,这在以后的服务与服务之间相互调用一般都是根据这个name 。

    eureka:

      client:

        serviceUrl:

          defaultZone: http://localhost:8761/eureka/

    server:

      port: 8762

    spring:

      application:

        name: service-hi

    Spring cloud有两种服务调用方式,一种是ribbon+restTemplate,另一种是feign。

    ribbon是一个负载均衡客户端,可以很好的控制htt和tcp的一些行为。Feign默认集成了ribbon。

    工程的配置文件指定服务的注册中心地址为  eureka的访问地址

    eureka:

      client:

        serviceUrl:

          defaultZone: http://localhost:8761/eureka/

    server:

      port: 8764

    spring:

      application:

        name: service-ribbon

    在工程的启动类中,通过@EnableDiscoveryClient向服务中心注册;并且向程序的ioc注入一个bean: restTemplate;

    通过@LoadBalanced注解表明这个restRemplate开启负载均衡的功能。

    在ribbon中它会根据服务名来选择具体的服务实例,根据服务实例在请求的时候会用具体的url替换掉服务名

    通过调用restTemplate.getForObject()方法调用业务

    Feign是一个声明式的伪Http客户端,它使得写Http客户端变得更简单。使用Feign,只需要创建一个接口并注解。它具有可插拔的注解特性,可使用Feign 注解和JAX-RS注解。Feign支持可插拔的编码器和解码器。Feign默认集成了Ribbon,并和Eureka结合,默认实现了负载均衡的效果。

    简而言之:

    Feign 采用的是基于接口的注解

    Feign 整合了ribbon

    eureka:

      client:

        serviceUrl:

          defaultZone: http://localhost:8761/eureka/

    server:

      port: 8765

    spring:

      application:

        name: service-feign

    在程序的启动类加上@EnableFeignClients注解开启Feign的功能:

    定义一个feign接口,通过@ FeignClient(“服务名”),来指定调用哪个服务。

    断路器(Hystrix)

    在微服务架构中,根据业务来拆分成一个个的服务,服务与服务之间可以相互调用(RPC),在Spring Cloud可以用RestTemplate+Ribbon和Feign来调用。为了保证其高可用,单个服务通常会集群部署。由于网络原因或者自身的原因,服务并不能保证100%可用,如果单个服务出现问题,调用这个服务就会出现线程阻塞,此时若有大量的请求涌入,Servlet容器的线程资源会被消耗完毕,导致服务瘫痪。服务与服务之间的依赖性,故障会传播,会对整个微服务系统造成灾难性的严重后果,这就是服务故障的“雪崩”效应。

    etflix开源了Hystrix组件,实现了断路器模式,SpringCloud对这一组件进行了整合。

    较底层的服务如果出现故障,会导致连锁故障。当对特定的服务的调用的不可用达到一个阀值(Hystric 是5秒20次) 断路器将会被打开。

    断路打开后,可用避免连锁故障,fallback方法可以直接返回一个固定值。

    在程序的启动类加@EnableHystrix注解开启Hystrix:

    在Service方法上加上@HystrixCommand注解。该注解对该方法创建了熔断器的功能,并指定了fallbackMethod熔断方法

    Feign中使用断路器

    Feign是自带断路器的,在D版本的Spring Cloud中,它没有默认打开。需要在配置文件中配置打开它,在配置文件加以下代码:

    feign.hystrix.enabled=true

    需要在FeignClient的Service接口的注解中加上fallback的指定类就行了

    基于service-ribbon 改造

    在主程序启动类中加入@EnableHystrixDashboard注解,开启hystrixDashboard:

    路由网关(zuul)

    在SpringCloud微服务系统中,一种常见的负载均衡方式是,客户端的请求首先经过负载均衡(zuul、Ngnix),再到达服务网关(zuul集群)

    ,然后再到具体的服务统一注册到高可用的服务注册中心集群,服务的所有的配置文件由配置服务管理(下一篇文章讲述),配置服务的配置文件放在git仓库,方便开发人员随时改配置。

    Zuul的主要功能是路由转发和过滤器。路由功能是微服务的一部分,比如/api/user转发到到user服务,/api/shop转发到到shop服务。zuul默认和Ribbon结合实现了负载均衡的功能。

    在入口applicaton类加上注解@EnableZuulProxy,开启zuul的功能:

    eureka:

      client:

        serviceUrl:

          defaultZone: http://localhost:8761/eureka/

    server:

      port: 8769

    spring:

      application:

        name: service-zuul

    zuul:

      routes:

        api-a:

          path: /api-a/**

          serviceId: service-ribbon

        api-b:

          path: /api-b/**

          serviceId: service-feign

    首先指定服务注册中心的地址为http://localhost:8761/eureka/,服务的端口为8769,服务名为service-zuul;以/api-a/ 开头的请求都转发给service-ribbon服务;以/api-b/开头的请求都转发给service-feign服务; 

    zuul不仅只是路由,并且还能过滤,做一些安全验证  

    @Component

    public class MyFilter extends ZuulFilter{

        private static Logger log = LoggerFactory.getLogger(MyFilter.class);

        @Override

        public String filterType() {

            return "pre";

        }

        @Override

        public int filterOrder() {

            return 0;

        }

        @Override

        public boolean shouldFilter() {

            return true;

        }

        @Override

        public Object run() {

            RequestContext ctx = RequestContext.getCurrentContext();

            HttpServletRequest request = ctx.getRequest();

            log.info(String.format("%s >>> %s", request.getMethod(), request.getRequestURL().toString()));

            Object accessToken = request.getParameter("token");

            if(accessToken == null) {

                log.warn("token is empty");

                ctx.setSendZuulResponse(false);

                ctx.setResponseStatusCode(401);

                try {

                    ctx.getResponse().getWriter().write("token is empty");

                }catch (Exception e){}

                return null;

            }

            log.info("ok");

            return null;

        }

    }

    分布式配置中心(Spring Cloud Config)

    在分布式系统中,由于服务数量巨多,为了方便服务配置文件统一管理,实时更新,所以需要分布式配置中心组件。在Spring Cloud中,有分布式配置中心组件spring cloud config ,它支持配置服务放在配置服务的内存中(即本地),也支持放在远程Git仓库中。在spring cloud config 组件中,分两个角色,一是config server,二是config client。

    Config Server

    在程序的入口Application类加上@EnableConfigServer注解开启配置服务器的功能

    需要在程序的配置文件application.properties文件配置以下:

    spring.application.name=config-server

    server.port=8888

    spring.cloud.config.server.git.uri=https://github.com/forezp/SpringcloudConfig/

    spring.cloud.config.server.git.searchPaths=respo

    spring.cloud.config.label=master

    spring.cloud.config.server.git.username=your username

    spring.cloud.config.server.git.password=your password

    ///参数详解

    spring.cloud.config.server.git.uri:配置git仓库地址

    spring.cloud.config.server.git.searchPaths:配置仓库路径

    spring.cloud.config.label:配置仓库的分支

    spring.cloud.config.server.git.username:访问git仓库的用户名

    spring.cloud.config.server.git.password:访问git仓库的用户密码

    config client

    spring.application.name=config-client

    spring.cloud.config.label=master

    spring.cloud.config.profile=dev

    spring.cloud.config.uri= http://localhost:8888/

    server.port=8881

    相关文章

      网友评论

        本文标题:Spring cloud 简单实现

        本文链接:https://www.haomeiwen.com/subject/ifqizxtx.html