DView项目问题总结

2018-07-11  本文已影响0人  晓序东风

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

通过 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

上一篇下一篇

猜你喜欢

热点阅读