DView项目问题总结

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

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

推荐阅读更多精彩内容