Hbase 热点(数据倾斜)问题解决方案---rowkey散列和预分区设计

Hbase的表会被划分为1....n个Region,被托管在RegionServer中。Region二个重要的属性:Startkey与EndKey表示这个Region维护的rowkey的范围,当我们要读写数据时,如果rowkey落在某个start-end key范围内,那么就会定位到目标region并且读写到相关的数据。

    默认情况下,当我们通过hbaseAdmin指定TableDescriptor来创建一张表时,只有一个region正处于混沌时期,start-end key无边界,可谓海纳百川。所有的rowkey都写入到这个region里,然后数据越来越多,region的size越来越大时,大到一定的阀值,hbase就会将region一分为二,成为2个region,这个过程称为分裂(region-split)。

    如果我们就这样默认建表,表里不断的put数据,更严重的是我们的rowkey还是顺序增大的,是比较可怕的。存在的缺点比较明显:首先是热点写,我们总是向最大的start key所在的region写数据,因为我们的rowkey总是会比之前的大,并且hbase的是按升序方式排序的。所以写操作总是被定位到无上界的那个region中;其次,由于热点,我们总是往最大的start key的region写记录,之前分裂出来的region不会被写数据,有点打入冷宫的感觉,他们都处于半满状态,这样的分布也是不利的。

    如果在写比较频繁的场景下,数据增长太快,split的次数也会增多,由于split是比较耗费资源的,所以我们并不希望这种事情经常发生。

    在集群中为了得到更好的并行性,我们希望有好的load blance,让每个节点提供的请求都是均衡的,我们也不希望,region不要经常split,因为split会使server有一段时间的停顿,如何能做到呢?

    随机散列与预分区二者结合起来,是比较完美的。预分区一开始就预建好了一部分region,这些region都维护着自己的start-end keys,在配合上随机散列,写数据能均衡的命中这些预建的region,就能解决上面的那些缺点,大大提供性能。

一、解决思路

    提供两种思路:hash与partition。

1、hash方案

    hash就是rowkey前面由一串随机字符串组成,随机字符串生成方式可以由SHA或者MD5方式生成,只要region所管理的start-end keys范围比较随机,那么就可以解决写热点问题。例如:

Java代码   收藏代码
  1. long currentId = 1L;  
  2. byte [] rowkey = Bytes.add(MD5Hash.getMD5AsHex(Bytes.toBytes(currentId))  
  3.                     .substring(08).getBytes(),Bytes.toBytes(currentId));  

     假如rowkey原本是自增长的long型,可以将rowkey转为hash再转为bytes,加上本身id转为bytes,这样就生成随便的rowkey。那么对于这种方式的rowkey设计,如何去进行预分区呢?

  1. 取样,先随机生成一定数量的rowkey,将取样数据按升序排序放到一个集合里。
  2. 根据预分区的region个数,对整个集合平均分割,即是相关的splitkeys。
  3. HBaseAdmin.createTable(HTableDescriptor tableDescriptor,byte[][] splitkeys)可以指定预分区的splitkey,即指定region间的rowkey临界值。

    创建split计算器,用于从抽样数据生成一个比较合适的splitkeys

Java代码   收藏代码
  1. public class HashChoreWoker implements SplitKeysCalculator{  
  2.     //随机取机数目  
  3.     private int baseRecord;  
  4.     //rowkey生成器  
  5.     private RowKeyGenerator rkGen;  
  6.     //取样时,由取样数目及region数相除所得的数量.  
  7.     private int splitKeysBase;  
  8.     //splitkeys个数  
  9.     private int splitKeysNumber;  
  10.     //由抽样计算出来的splitkeys结果  
  11.     private byte[][] splitKeys;  
  12.   
  13.     public HashChoreWoker(int baseRecord, int prepareRegions) {  
  14.         this.baseRecord = baseRecord;  
  15.         //实例化rowkey生成器  
  16.         rkGen = new HashRowKeyGenerator();  
  17.         splitKeysNumber = prepareRegions - 1;  
  18.         splitKeysBase = baseRecord / prepareRegions;  
  19.     }  
  20.   
  21.     public byte[][] calcSplitKeys() {  
  22.         splitKeys = new byte[splitKeysNumber][];  
  23.         //使用treeset保存抽样数据,已排序过  
  24.         TreeSet<byte[]> rows = new TreeSet<byte[]>(Bytes.BYTES_COMPARATOR);  
  25.         for (int i = 0; i < baseRecord; i++) {  
  26.             rows.add(rkGen.nextId());  
  27.         }  
  28.         int pointer = 0;  
  29.         Iterator<byte[]> rowKeyIter = rows.iterator();  
  30.         int index = 0;  
  31.         while (rowKeyIter.hasNext()) {  
  32.             byte[] tempRow = rowKeyIter.next();  
  33.             rowKeyIter.remove();  
  34.             if ((pointer != 0) && (pointer % splitKeysBase == 0)) {  
  35.                 if (index < splitKeysNumber) {  
  36.                     splitKeys[index] = tempRow;  
  37.                     index ++;  
  38.                 }  
  39.             }  
  40.             pointer ++;  
  41.         }  
  42.         rows.clear();  
  43.         rows = null;  
  44.         return splitKeys;  
  45.     }  
  46. }  

     KeyGenerator及实现

Java代码   收藏代码
  1. //interface  
  2. public interface RowKeyGenerator {  
  3.     byte [] nextId();  
  4. }  
  5. //implements  
  6. public class HashRowKeyGenerator implements RowKeyGenerator {  
  7.     private long currentId = 1;  
  8.     private long currentTime = System.currentTimeMillis();  
  9.     private Random random = new Random();  
  10.     public byte[] nextId() {  
  11.         try {  
  12.             currentTime += random.nextInt(1000);  
  13.             byte[] lowT = Bytes.copy(Bytes.toBytes(currentTime), 44);  
  14.             byte[] lowU = Bytes.copy(Bytes.toBytes(currentId), 44);  
  15.             return Bytes.add(MD5Hash.getMD5AsHex(Bytes.add(lowU, lowT)).substring(08).getBytes(),  
  16.                     Bytes.toBytes(currentId));  
  17.         } finally {  
  18.             currentId++;  
  19.         }  
  20.     }  
  21. }  

     unit test case测试

Java代码   收藏代码
  1. @Test  
  2. public void testHashAndCreateTable() throws Exception{  
  3.         HashChoreWoker worker = new HashChoreWoker(1000000,10);  
  4.         byte [][] splitKeys = worker.calcSplitKeys();  
  5.           
  6.         HBaseAdmin admin = new HBaseAdmin(HBaseConfiguration.create());  
  7.         TableName tableName = TableName.valueOf("hash_split_table");  
  8.           
  9.         if (admin.tableExists(tableName)) {  
  10.             try {  
  11.                 admin.disableTable(tableName);  
  12.             } catch (Exception e) {  
  13.             }  
  14.             admin.deleteTable(tableName);  
  15.         }  
  16.   
  17.         HTableDescriptor tableDesc = new HTableDescriptor(tableName);  
  18.         HColumnDescriptor columnDesc = new HColumnDescriptor(Bytes.toBytes("info"));  
  19.         columnDesc.setMaxVersions(1);  
  20.         tableDesc.addFamily(columnDesc);  
  21.   
  22.         admin.createTable(tableDesc ,splitKeys);  
  23.   
  24.         admin.close();  
  25.     }  

     查看建表结果,执行:scan 'hbase:meta'


    以上我们只是显示了部分region的信息,可以看到region的start-end key还是比较随机散列的。同样可以查看hdfs的目录结构,的确和预期的38个预分区一致: 


    以上就是按照hash方式,预建好分区,以后再插入数据的时候,也是按照此rowkeyGenerator的方式生成rowkey。

2、partition的方式

    partition顾名思义就是分区式,这种分区有点类似于mapreduce中的partitioner,将区域用长整数作为分区号,每个region管理着相应的区域数据,在rowkey生成时,将ID取模后,然后拼上ID整体作为rowkey,这个比较简单,不需要取样,splitkeys也非常简单,直接是分区号即可。直接上代码:

Java代码   收藏代码
  1. public class PartitionRowKeyManager implements RowKeyGenerator,  
  2.         SplitKeysCalculator {  
  3.   
  4.     public static final int DEFAULT_PARTITION_AMOUNT = 20;  
  5.     private long currentId = 1;  
  6.     private int partition = DEFAULT_PARTITION_AMOUNT;  
  7.     public void setPartition(int partition) {  
  8.         this.partition = partition;  
  9.     }  
  10.   
  11.     public byte[] nextId() {  
  12.         try {  
  13.             long partitionId = currentId % partition;  
  14.             return Bytes.add(Bytes.toBytes(partitionId),  
  15.                     Bytes.toBytes(currentId));  
  16.         } finally {  
  17.             currentId++;  
  18.         }  
  19.     }  
  20.   
  21.     public byte[][] calcSplitKeys() {  
  22.         byte[][] splitKeys = new byte[partition - 1][];  
  23.         for(int i = 1; i < partition ; i ++) {  
  24.             splitKeys[i-1] = Bytes.toBytes((long)i);  
  25.         }  
  26.         return splitKeys;  
  27.     }  
  28. }  

    calcSplitKeys方法比较单纯,splitkey就是partition的编号,测试类如下:

Java代码   收藏代码
  1. @Test  
  2.     public void testPartitionAndCreateTable() throws Exception{  
  3.           
  4.         PartitionRowKeyManager rkManager = new PartitionRowKeyManager();  
  5.         //只预建10个分区  
  6.         rkManager.setPartition(10);  
  7.           
  8.         byte [][] splitKeys = rkManager.calcSplitKeys();  
  9.           
  10.         HBaseAdmin admin = new HBaseAdmin(HBaseConfiguration.create());  
  11.         TableName tableName = TableName.valueOf("partition_split_table");  
  12.           
  13.         if (admin.tableExists(tableName)) {  
  14.             try {  
  15.                 admin.disableTable(tableName);  
  16.   
  17.             } catch (Exception e) {  
  18.             }  
  19.             admin.deleteTable(tableName);  
  20.         }  
  21.   
  22.         HTableDescriptor tableDesc = new HTableDescriptor(tableName);  
  23.         HColumnDescriptor columnDesc = new HColumnDescriptor(Bytes.toBytes("info"));  
  24.         columnDesc.setMaxVersions(1);  
  25.         tableDesc.addFamily(columnDesc);  
  26.   
  27.         admin.createTable(tableDesc ,splitKeys);  
  28.   
  29.         admin.close();  
  30.     }  

     同样我们可以看看meta表和hdfs的目录结果,其实和hash类似,region都会分好区。

     通过partition实现的loadblance写的话,当然生成rowkey方式也要结合当前的region数目取模而求得,大家同样也可以做些实验,看看数据插入后的分布。

     在这里也顺提一下,如果是顺序的增长型原id,可以将id保存到一个数据库,传统的也好,redis的也好,每次取的时候,将数值设大1000左右,以后id可以在内存内增长,当内存数量已经超过1000的话,再去load下一个,有点类似于oracle中的sqeuence.

     随机分布加预分区也不是一劳永逸的。因为数据是不断地增长的,随着时间不断地推移,已经分好的区域,或许已经装不住更多的数据,当然就要进一步进行split了,同样也会出现性能损耗问题,所以我们还是要规划好数据增长速率,观察好数据定期维护,按需分析是否要进一步分行手工将分区再分好,也或者是更严重的是新建表,做好更大的预分区然后进行数据迁移。如果数据装不住了,对于partition方式预分区的话,如果让它自然分裂的话,情况分严重一点。因为分裂出来的分区号会是一样的,所以计算到partitionId的话,其实还是回到了顺序写年代,会有部分热点写问题出现,如果使用partition方式生成主键的话,数据增长后就要不断地调整分区了,比如增多预分区,或者加入子分区号的处理.(我们的分区号为long型,可以将它作为多级partition)

    以上基本已经讲完了防止热点写使用的方法和防止频繁split而采取的预分区。但rowkey设计,远远也不止这些,比如rowkey长度,然后它的长度最大可以为char的MAXVALUE,但是看过之前我写KeyValue的分析知道,我们的数据都是以KeyValue方式存储在MemStore或者HFile中的,每个KeyValue都会存储rowKey的信息,如果rowkey太大的话,比如是128个字节,一行10个字段的表,100万行记录,光rowkey就占了1.2G+所以长度还是不要过长,另外设计,还是按需求来吧。

<think>好的,用户想了解HBase查询中的热点问题解决方案。首先,我需要回忆一下HBase热点问题是怎么回事。热点通常发生在某个Region Server处理大量请求,而其他节点负载较低,导致性能瓶颈。用户提到的引用里也提到了数据倾斜Rowkey设计问题。 用户的问题主要集中在查询时的热点,所以得先分析查询热点写入热点的区别。查询热点可能是因为Rowkey设计不合理,导致某些Region被频繁访问。例如,如果Rowkey是顺序递增的,比如时间戳,那么最新的数据可能集中在某个Region,导致查询压力大。 解决方案方面,引用里提到了预分区Rowkey散列化、加盐等方法。需要详细说明每种方法如何实施。比如,预分区可以在建表时划分多个Region,避免初始时只有一个Region。Rowkey设计部分可能需要举例,比如反转时间戳或者组合字段,确保分布均匀。 还要考虑列族列名的优化,因为过大的列名会增加存储传输开销,进而影响索引大小,引用2提到了这点。可能需要建议用户缩短列名或使用编码。 另外,用户可能没有意识到热点不仅出现在写入时,查询时同样会有。比如,频繁查询某个特定范围的Rowkey,导致某个Region负载过高。这时候可能需要结合业务逻辑调整查询模式,或者使用缓存机制。 需要检查是否有遗漏的解决方案,比如负载均衡机制是否开启,或者手动拆分Region。引用3中提到了默认情况下的Region行为,所以预分区避免单调递增Rowkey是关键。 最后,确保回答结构清晰,分点说明问题解决方案,并附上相关问题供用户进一步学习。可能还要提醒用户根据实际业务场景选择合适的方法,比如加盐可能会影响范围查询,需要权衡。</think>### HBase查询热点问题解决方案 #### 一、查询热点问题表现 当HBase表出现查询热点时,**特定Region的请求量显著高于其他Region**,通常伴随以下现象: 1. 单个Region Server的读写请求数比其他节点高2-10倍[^1] 2. UI监控显示某Region的请求量比其他Region高1-2个数量级[^1] 3. 客户端出现频繁的读超时或响应延迟 #### 二、核心原因分析 1. **Rowkey设计缺陷** - 单调递增Rowkey(如时间戳)导致新数据集中写入/查询同一Region[^3] - 固定前缀造成数据局部聚集(例如`user_1000_`开头的Rowkey集中存储) 2. **数据分布不均** -预分区的表初始只有一个Region,所有查询集中在单点[^3] - 未合理使用散列算法导致数据倾斜 3. **查询模式问题** - 高频访问特定业务数据(如热门商品ID) - 范围查询未优化导致扫描大量连续Rowkey #### 三、针对性解决方案 ##### 1. Rowkey优化(关键措施) | 方法 | 实现 | 适用场景 | 注意事项 | |------|------|----------|----------| | **散列化** | `MD5(Rowkey).substr(0,4)+原Rowkey` | 均匀分布需求 | 牺牲范围查询能力 | | **反转键** | `reverse(timestamp)+业务ID` | 时间序列数据 | 需配合前缀过滤 | | **组合键** | `业务字段1_业务字段2_散列值` | 多维度查询 | 需明确查询模式 | | **加盐值** | `(hash(业务字段)%region数)_原Rowkey` | 强制均匀分布 | 增加计算开销 | > 示例:对时间戳`20230801120000`优化为`00_200001108322`(反转+加盐) ##### 2. 预分区策略 ```java // 创建表时指定分区边界 byte[][] splits = new byte[][]{ Bytes.toBytes("A|"), Bytes.toBytes("M|"), Bytes.toBytes("S|") }; admin.createTable(tableDesc, splits); ``` - 建议根据**业务查询模式**设计分区 - 分区数可按`RegionServer数量 * 2`作为基准[^3] ##### 3. 查询模式优化 - 避免全表扫描:设置合理的`startRow``stopRow` - 使用协处理器将热点查询逻辑下沉到RegionServer - 对高频查询结果启用BlockCache(调整`hbase.blockcache.size`) ##### 4. 系统级调优 - 开启自动负载均衡:`balance_switch true` - 调整Region大小:`hbase.hregion.max.filesize=30GB`(默认10GB) - 启用压缩:`COMPRESSION => 'SNAPPY'`减少IO压力[^2] #### 四、诊断工具链 1. **HBase Shell** ```shell hbase> status 'detailed' # 查看Region分布 hbase> scan 'hbase:meta' # 检查Region边界 ``` 2. **HBase UI** - Regionserver的请求量监控 - Region级别的读写统计 3. **日志分析** - 关注`RegionTooBusyException` - 跟踪慢查询日志中的Rowkey模式 #### 五、典型场景解决方案 **场景:电商订单查询热点** - 问题:按订单创建时间查询产生时间范围热点 - 优化方案: 1. Rowkey设计:`reverse(timestamp)_userid_md5(orderid)` 2. 预分区按小时划分 3. 建立反向索引表用于用户维查询
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值