美文网首页
DView项目问题总结

DView项目问题总结

作者: 晓序东风 | 来源:发表于2018-07-11 22:38 被阅读0次

    DView是一款网管系统,基于Spring Boot开发,Maven构建,主要包含5个模块:

    • Probe:与网络设备交互的服务,执行具体的网络管理指令,如:更新网络设备软件、发现网络中的设备、监视网络设备等等
    • CoreServer:负责发送相关的网管指令到Probe,并处理Probe返回的相关数据,数据存储、查询等任务
    • DataServer:数据库服务,为CoreServer提供数据接口,使用MongoDB作为持久化数据库,Redis作为缓存数据库
    • WebServer:用户操作界面,用户进行的各种操作将会发送给CoreServe
    • Common:公共的数据结构、方法等

    通过 java -jar dataServer-0.0.1-SNAPSHOT.jar.jar --spring.config.location=application.properties外部化配置不生效?

    我们使用Docker部署Probe、CoreServer、DataServer、WebSiteServer,考虑到部署的时候需要根据具体情况修改配置文件,所有配置文件不能放在classpath中,就将配置文件copy一份放在和JAR同一级的目录下,并在Docker中通过java -jar dataServer-0.0.1-SNAPSHOT.jar.jar --spring.config.location=application.properties来指定配置文件的路径,我们的初衷是利用Spring Boot加载配置文件的优先级机制,通过显示指定配置文件,从而实现配置的外部化,但是,当通过上面那种方式启动时发现程序运行的时候始终使用的是classpath下的application.properties,而不是用我们显示指定的spring.config.location=application.properties文件
    难道Spring Boot加载的配置文件的优先级不是:当前目录下的/config子目录>当前目录>classpath下的/config包>classpath根路径?
    等等,先来看看DataServer中配置文件的使用方式,如下:

    application-dev.properties
    application-prod.properties
    application.properties
    

    为了应对多环境,我们使用了Profiles,在application.properties指定内容如下:

    spring.profiles.active=prod
    

    我们错误的认为在启动的时候指定了配置文件的路径,Spring Boot就会加载并使用这个配置文件,而忽略其它存在的配置文件
    后来通过查阅资料和阅读源码,发现spring.config.location=application.properties仅仅是为Spring Boot添加了额外的属性文件搜索路径,当classpath路径下存application.properties文件时,无论如何,Spring Boot都会先加载该配置文件,如果其中指定了profiles,之后Spring Boot会加载profiles对应的配置文件,我们在application.properties指定了spring.profiles.active=prod,所有Spring Boot会查找名为application-prod.properties的配置文件,但是因为我们在spring.config.location中给定的值是application.properties,所以对于Spring Boot而言,application-prod.properties配置文件只存在于classpath下,故它每次加载并使用的就是classpath下的application-prod.properties,所有,是我们在spring.config.location给定的名字错了,修改为spring.config.location=application-prod.properties,即可(注意:放在与JAR同级目录下的属性配置文件名也要改成application-prod.properties)

    使用RateLimiter实现限流出现的问题

    在高并发系统中一般会采用3种方式来保护系统:
    1.缓存:可以提升系统的访问速度和增大系统处理容量
    2.降级:当服务出现问题或者影响到系统的核心流程时,需要暂时屏蔽掉相关服务
    3.限流:通过限制并发访问速率,当达到限制速率时就采取拒绝服务、排队或降级等策略
    RateLimiter是Google开源工具包Guava提供的一个限流工具类,基于令牌桶算法
    在DataServer中,使用RateLimiter实现了简单的对RESTful API访问限速机制
    POM文件中加入如下依赖

    <dependency>
        <groupId>com.google.guava</groupId>
        <artifactId>guava</artifactId>
        <version>19.0</version>
    </dependency>
    

    考虑到所有的RESTful API访问都要进行限流,所以基于Filter实现,在Filter中实现限流的逻辑,因为所有请求都要经过Filter,代码很简单

    @Component
    public class APICallRateLimiter implements Filter {
        
        /**
         * 限流大小
         */
        @Value("${api.restful.rate.limit}")
        private double rateLimit;
        
        /**
         * 等待时间
         */
        @Value("${api.restful.wait.time}")
        private long waitTime;
        
        public void setWaitTime(long waitTime) {
            this.waitTime = waitTime;
        }
        
        @Override
        public void destroy() {
            
        }
    
        @Override
        public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
            throws IOException, ServletException {
            //使用限制,允许的并发请求
            RateLimiter rateLimiter = RateLimiter.create(this.rateLimit);
            //阻塞式
            //rateLimiter.acquire();
            //非阻塞式,允许等待的时间
            if (rateLimiter.tryAcquire(this.waitTime, TimeUnit.MILLISECONDS)) {
                chain.doFilter(request, response);
            }
        }
    
        @Override
        public void init(FilterConfig config) throws ServletException {
            
        }
    }
    

    注册我们上面自定义的APICallRateLimiter到Spring中

    @Configuration
    public class APIConfig {
        
        /**
         * 限制的URL
         */
        @Value("${api.restful.limit.url}")
        private String limitURL;
        
        /**
         * Description: 配置过滤器到Spring容器<br> 
         *  
         * @author GC<br>
         * @return <br>
         */
        @Bean
        public FilterRegistrationBean<APICallRateLimiter> apiFilter() {
            FilterRegistrationBean<APICallRateLimiter> registrationBean = new FilterRegistrationBean<APICallRateLimiter>();
            registrationBean.setFilter(new APICallRateLimiter());
            List<String> urls = new ArrayList<String>();
            urls.add(this.limitURL);
            registrationBean.setUrlPatterns(urls);
            return registrationBean;
        }
    }
    

    主要碰到了如下几个问题:
    1.APICallRateLimiter类中不能通过@Value方式获取配置文件中的api.restful.rate.limit、api.restful.wait.time等值
    2.RateLimiter并没有生效,没有起到限流的作用
    第1个问题是因为默认情况下Filter是不被Spring自动管理的,所以获取不到属性配置文件中的值,为了能够获取属性配置文件中的值,可以将我们自定义的Filter交由Spring管理,很简单,定义一个创建自定义Filter的方法,然后在该方法上添加@Bean
    第2个问题算是实现逻辑上的问题,因为每次请求都会触发doFilter,所以RateLimiter每次也会被重新创建,对于每一次请求而言都是使用的不同的RateLimiter,所以需要将RateLimiter变成声明一次,多次使用
    最终修改代码如下:

    @Component
    public class APICallRateFilter implements Filter {
        
        /**
         * 等待时间
         */
        private long waitTime;
        
        /**
         * RateLimiter
         */
        @Autowired
        private RateLimiter rateLimiter;
        
        public void setWaitTime(long waitTime) {
            this.waitTime = waitTime;
        }
        
        @Override
        public void destroy() {
            
        }
    
        @Override
        public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
            throws IOException, ServletException {
            //使用限制,允许的并发请求
            //阻塞式
            //rateLimiter.acquire();
            //非阻塞式,允许等待的时间
            if (this.rateLimiter.tryAcquire(this.waitTime, TimeUnit.MILLISECONDS)) {
                chain.doFilter(request, response);
            }
        }
    
        @Override
        public void init(FilterConfig config) throws ServletException {
            
        }
    }
    
    @Configuration
    public class WebConfig {
        
        /**
         * 限流大小
         */
        @Value("${api.restful.rate.limit}")
        private double rateLimit;
        
        /**
         * 等待时间
         */
        @Value("${api.restful.wait.time}")
        private long waitTime;
        
        /**
         * 限制的URL
         */
        @Value("${api.restful.limit.url}")
        private String limitURL;
        
        /**
         * Description: 创建APIFilter<br> 
         *  
         * @author GC <br>
         * @return APIFilter <br>
         */
        @Bean
        public APICallRateFilter createAPIFilter() {
            APICallRateFilter apiFilter = new APICallRateFilter();
            apiFilter.setWaitTime(waitTime);
            return apiFilter;
        }
        
        /**
         * Description: 创建RateLimiter<br> 
         *  
         * @author GC<br>
         * @return RateLimiter<br>
         */
        @Bean
        public RateLimiter createRateLimiter() {
            RateLimiter rateLimiter = RateLimiter.create(this.rateLimit);
            return rateLimiter;
        }
        
        /**
         * Description: 配置过滤器到Spring容器<br> 
         *  
         * @author GC<br>
         * @return <br>
         */
        @Bean
        public FilterRegistrationBean<APICallRateFilter> apiFilter() {
            FilterRegistrationBean<APICallRateFilter> registrationBean = new FilterRegistrationBean<APICallRateFilter>();
            registrationBean.setFilter(createAPIFilter());
            List<String> urls = new ArrayList<String>();
            urls.add(this.limitURL);
            registrationBean.setUrlPatterns(urls);
            return registrationBean;
        }
    }
    

    使用Apache开源的压力测试工具JMeter进行测试,设置1s启动500个线程模拟500/s的并发请求,其中RateLimiter的相关参数如下,测试结果如下图所示

    #Current limiting setting
    api.restful.limit.url=/api/db/*
    api.restful.rate.limit=100
    api.restful.wait.time=5000
    
    JMeter测试结果

    可以看到整个测试结果表明,对于500个并发请求,通过使用RateLimiter控制后,系统处理完500个请求总共耗时4s

    相关文章

      网友评论

          本文标题:DView项目问题总结

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