ribbon源码阅读--IPing源码分析

本人小白,这个文章是本人的阅读笔记,不是权威解读,需要自己甄别对错

最后一个部分了,就是IPing组件了,接着看看这个

IPing机制在我的理解当中可能就是就是弥补Eureka的服务信息变更的延时效应的。

我们算算,假设一个服务宕机了,其他服务Eureka Client列表最慢获取到服务变更消息的时间,也是复习下Eureka的相关知识

  • 服务故障移除 30S执行一次,但是的但是,服务租约的过期是需要2 * 90S,就是3分钟,取最长的,那么需要6次检测才能感知故障进行移除。
  • 故障实例移除了,但是只是清空了注册表和读写缓存,只读缓存需要花费30S进行同步,才能从最近变更队列中获取。
  • Eureka Client是花费30S进行一次增量获取

假设啊,时间都错开了,那么SumTime = 180 + 30 + 30 大概需要4分钟,并且Ribbon的AllList从Eureka Client本地注册表拉取也是30S进行一次,那么感知一个服务宕机了,需要花费好几分钟,这不是有点小坑?

这种情况下,在并发比较高的场景,那么请求宕机服务的线程都会超时,超时造成的当前线程资源耗尽会引发服务雪崩效应,不知道是不是基于这个考虑,Hystrix中资源隔离和服务熔断降级很好的解决了这个情况!

IPing机制在一定情况下也缓解了这种情况发生。

说说主要的IPing实现吧

首先IPing是怎么实例化的呢?

    @Bean
    @ConditionalOnMissingBean
    public IPing ribbonPing(IClientConfig config) {
        if (this.propertiesFactory.isSet(IPing.class, serviceId)) {
            return this.propertiesFactory.get(IPing.class, config, serviceId);
        }
        NIWSDiscoveryPing ping = new NIWSDiscoveryPing();
        ping.initWithNiwsConfig(config);
        return ping;
    }

如果配置了自定义IPing机制就用自己那一套,没配就走NIWSDiscoveryPing

那么在哪进行使用的呢,在ZoneAwareLoadBalancer进行初始化的时候传入了,向上追踪父类构造,你会发现在BaseLoadBalancer初始化时候使用了,顺便说一嘴,BaseLoadBalancer是原生的Ribbon调用接口类。

        String clientName = clientConfig.getClientName();
        this.name = clientName;
        // 默认定时调度30S
        int pingIntervalTime = Integer.parseInt(""
                + clientConfig.getProperty(
                        CommonClientConfigKey.NFLoadBalancerPingInterval,
                        Integer.parseInt("30")));
        int maxTotalPingTime = Integer.parseInt(""
                + clientConfig.getProperty(
                        CommonClientConfigKey.NFLoadBalancerMaxTotalPingTime,
                        Integer.parseInt("2")));
        //启动IPing机制
        setPingInterval(pingIntervalTime);
        setMaxTotalPingTime(maxTotalPingTime);

        // cross associate with each other
        // i.e. Rule,Ping meet your container LB
        // LB, these are your Ping and Rule guys ...
        setRule(rule);
        setPing(ping);

接着会触发一个定时任务

    void setupPingTask() {
        if (canSkipPing()) {
            return;
        }
        if (lbTimer != null) {
            lbTimer.cancel();
        }
        lbTimer = new ShutdownEnabledTimer("NFLoadBalancer-PingTimer-" + name, true);
        //30S执行一次        
        lbTimer.schedule(new PingTask(), 0, pingIntervalSeconds * 1000);
        forceQuickPing();
    }

任务核心执行方法

    public void runPinger() throws Exception {
            if (!pingInProgress.compareAndSet(false, true)) { 
                return; // Ping in progress - nothing to do
            }
            
            // we are "in" - we get to Ping

            Server[] allServers = null;
            boolean[] results = null;

            Lock allLock = null;
            Lock upLock = null;

            try {
                /*
                 * The readLock should be free unless an addServer operation is
                 * going on...
                 */
                allLock = allServerLock.readLock();
                allLock.lock();
                allServers = allServerList.toArray(new Server[allServerList.size()]);
                allLock.unlock();

                int numCandidates = allServers.length;
                //使用IPing的isAlive()方法判断服务是否可用
                results = pingerStrategy.pingServers(ping, allServers);

                final List<Server> newUpList = new ArrayList<Server>();
                final List<Server> changedServers = new ArrayList<Server>();

                for (int i = 0; i < numCandidates; i++) {
                    boolean isAlive = results[i];
                    Server svr = allServers[i];
                    boolean oldIsAlive = svr.isAlive();
                    //重置服务状态
                    svr.setAlive(isAlive);

                    if (oldIsAlive != isAlive) {
                        //加入服务状态改变列表
                        changedServers.add(svr);
                        logger.debug("LoadBalancer [{}]:  Server [{}] status changed to {}", 
                            name, svr.getId(), (isAlive ? "ALIVE" : "DEAD"));
                    }

                    if (isAlive) {
                        newUpList.add(svr);
                    }
                }
                upLock = upServerLock.writeLock();
                upLock.lock();
                upServerList = newUpList;
                upLock.unlock();
                //通知服务状态变更
                notifyServerStatusChangeListener(changedServers);
            } finally {
                pingInProgress.set(false);
            }
        }
    }

就是这一块调用了,目的也是重置服务状态

具体的实现类我们看几个主要的,不常用的就懒得看了

NIWSDiscoveryPing

默认实现

    public boolean isAlive(Server server) {
            boolean isAlive = true;
            if (server!=null && server instanceof DiscoveryEnabledServer){
                DiscoveryEnabledServer dServer = (DiscoveryEnabledServer)server;                
                InstanceInfo instanceInfo = dServer.getInstanceInfo();
                if (instanceInfo!=null){                    
                    InstanceStatus status = instanceInfo.getStatus();
                    if (status!=null){
                        isAlive = status.equals(InstanceStatus.UP);
                    }
                }
            }
            return isAlive;
        }

从本地注册表拉取判断服务实例状态,但是有个疑问,Ribbon的AllList拉取,即enableAndInitLearnNewServersFeature()启动的定时更新AllList任务也是30S执行一次,这个任务是全量更新服务列表,那么这个IPing的作用是???不懂,待大佬解答啊

PingUrl

    public boolean isAlive(Server server) {
                String urlStr   = "";
                if (isSecure){
                    urlStr = "https://";
                }else{
                    urlStr = "http://";
                }
                urlStr += server.getId();
                urlStr += getPingAppendString();

                boolean isAlive = false;

                HttpClient httpClient = new DefaultHttpClient();
                HttpUriRequest getRequest = new HttpGet(urlStr);
                String content=null;
                try {
                    HttpResponse response = httpClient.execute(getRequest);
                    content = EntityUtils.toString(response.getEntity());
                    isAlive = (response.getStatusLine().getStatusCode() == 200);
                    if (getExpectedContent()!=null){
                        LOGGER.debug("content:" + content);
                        if (content == null){
                            isAlive = false;
                        }else{
                            if (content.equals(getExpectedContent())){
                                isAlive = true;
                            }else{
                                isAlive = false;
                            }
                        }
                    }
                } catch (IOException e) {
                    e.printStackTrace();
                }finally{
                    // Release the connection.
                    getRequest.abort();
                }

                return isAlive;
        }

就是直接请求服务了,默认的就是http://192.168.31.128:8080,默认情况下返回200就任务服务可用,当然可配置接口和返回值进行二次验证

这个机制对Eureka Client服务感知延迟做了一定的保障,但是这个是不是有点耗费性能

好了,IPing机制就说这点,没啥好说的

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

推荐阅读更多精彩内容