zoukankan      html  css  js  c++  java
  • HBase写被block的分析

    一个线上集群出现莫名奇妙不能写入数据的bug,log中不断打印如下信息:
    引用
    2011-11-09 07:35:45,911 INFO org.apache.hadoop.hbase.regionserver.HRegion: Blocking updates for 'IPC Server handler 32 on 60020' on
    region xxx,333-2395000000032117,1320773734010.9a7ae39b5a42ccfa1fa6118aa8f79195.: memstore size 128.0m is >= than blocking 128.0m size

        我们知道每次put时会检查当前的memstore大小,当大于flush值的一个系数时(系数默认为2倍),就会block住这次写请求,并提交一个flush任务。但是很奇怪的是,用户此时再也不能往这个region写数据了,并在大约10多个小时以后又神奇地自然恢复了。
        原因是什么呢?
        经过一番检查,发现了hbase的一个bug,我们准备修改后提交到社区,不过因为实在太有趣了,体现了分布式事务的很有趣特征,所以先在此分享一下原因吧。
        这个问题是由以下四个事件共同组成的,我把代码简单化后作如下整理:
    1 put:
    Java代码  收藏代码
    1. put{  
    2.   checkResources{  
    3.     while (this.memstoreSize.get() > this.blockingMemStoreSize) {  
    4.       if(flushRequested==true)  
    5.         continue;  
    6.       flushRequested = true;  
    7.       flushQueue.add(this);  
    8.     }  
    9. ...  
    10.   }  
    11. ...  
    12. }  


    2 memstoreFlusher:
    Java代码  收藏代码
    1. while(!serverstop){  
    2.   task = flushQueue.poll();  
    3.   if(task == null)  
    4.     continue;  
    5.   if(closing)  
    6.     continue;  
    7.   try{  
    8.     if(closed)  
    9.       continue;  
    10.     if(flush(task))  
    11.       continue;  
    12.     else  
    13.       break;  
    14.   }finally{  
    15.     flushRequested = false;  
    16.   }  
    17. }  


    3 split:
    Java代码  收藏代码
    1. ...  
    2. closing = true;  
    3. closed = true;  
    4. ...  


    4 rollback:
    Java代码  收藏代码
    1. ...  
    2. closing = false;  
    3. closed = false;  
    4. ...  



        故障还原:当该region执行一次flush时,flushRequested被put线程置为了true,并push一个flush任务。然后memstoreFlusher检查到该任务时,刚好split开始进行,进行到了CLOSED_PARENT_REGION那一步,处于closing状态,于是memstoreflusher跳过任务,但在这里,memstoreflusher仍然报告该任务完成了,于是flush队列被清空。
        但split在执行splitStoreFiles时,因为hdfs的问题失败了(具体原因是namenode在close一个文件的时候失败,不停地retry并超时),此时split开始执行回滚,即该region恢复到split之前的状态,于是我们发现该region又重新onlined。

        虽然split在rollback的时候会将closing和closed状态置回来,但因为flush队列己然被清空了,于是陷入以下循环:
    • put数据的线程,发现需要flush,但flushRequested为true,说明还有flush任务没完成,于是继续等待,并不会提交flush任务
    • memstoreFlush的线程,每次取flushQueue都为空,于是循环等待put线程提交flush任务,因此写数据就被block住了


        以上悲催的情况将一直持续,直到迎来cleanOldLogs任务。因为cleanOldLogs会每小时执行一次,它会将最早的.logs目录下的文件移到.oldlogs目录下,但移之前先检查该文件中所有的数据是否己经flush到磁盘了,如果还没有就将该region执行一次flush。所以在经过n小时以后,.logs终于滚动到了用户之前卡住的那一段,这时就强制执行flush任务,因此flushQueue队列就不为空了,死循环被打破。系统也就自愈了。
  • 相关阅读:
    Java 匿名内部类
    【嘎】数组-搜索插入位置
    【嘎】数组-1266. 访问所有点的最小时间
    【嘎】数组-1313. 解压缩编码列表
    【嘎】数组-1431. 拥有最多糖果的孩子
    element-ui下拉多选报错Error in event handler for "handleOptionClick": "TypeError: value.push is not a function"
    【嘎】数组-有效的山脉数组
    【嘎】数组-打家劫舍
    【嘎】字符串-字符串中的第一个唯一字符
    linux
  • 原文地址:https://www.cnblogs.com/cl1024cl/p/6205194.html
Copyright © 2011-2022 走看看