文章目录

  • 1. 为什么使用分区表?
  • 2. 分区表DDL
  • 2.1 创建分区表
  • 2.2 增加分区
  • 2.3 删除分区
  • 2.4 重命名分区
  • 2.5 修复分区
  • 2.6 修改分区
  • 3. 分区表的数据导入
  • (1) 静态分区
  • (2) 动态分区
  • 4. 查询
  • 4.1 查询分区表数据
  • 4.2 查询分区表结构
  • 5. 小结


1. 为什么使用分区表?

  • 条件:假如现有一个角色表t_all_hero,该表中有6个清洗干净的互不干扰的数据文件:射手、坦克、战士、法师、刺客、辅助
  • 要求:查找出名字为射手且生命值大于6000的角色人数
  • 惯性解决方法:按照MySQL思维很容易想到
  1. 问:如何提高效率?这样虽然能够解决问题,但是由于要进行全表扫描,效率非常低。
  2. 答:由于6个文件已经清洗好了,且互不干扰,所以我们只需要从archer.txt中进行扫描就可以了。

总结:为了避免查询时进行全表扫描,Hive可以根据指定的字段对表进行分区扫描,提高查询效率。

2. 分区表DDL

2.1 创建分区表

在创建分区表时选择partitioned by关键字。比如:

create table person(
	id int,
	name string comment '名字',
	sex string comment '性别',
	city string comment '城市'
) 
-- 分区字段可以是1个,也可以是多个
partitioned by (sex_part string, city_part string)

注意:

  1. 分区字段不能是表中已经存在的字段,开发中一般是对某字段取别名做为分区字段。
  2. 分区字段最终也会以虚拟字段的形式显示在表结构上。如果以select * from person;,则查询结果在表字段name sexcity的基础上多显示分区字段sex_part city_part
  3. 分区字段本质上是将打上相同分区标签的数据放在同一个文件夹下,利用文件夹来区分不同数据。
  1. 分区字段可以是1个,也可以是多个。

2.2 增加分区

  • add partition增加分区,并不会自动加载数据。如果分区位置中不存在数据,查询时将不会返回结果。因此需要保证增加的分区位置路径下,数据已经存在,或者增加完分区之后导入分区数据。
-- 1. 一次增加多个分区(这样算是静态设置分区)
ALTER TABLE table_name ADD PARTITION (dt='2008-08-08', country='us') location '/path/to/us/part080808'
PARTITION (dt='2008-08-09', country='us') location '/path/to/us/part080809';

-- 2. 再使用load data 或 hadoop dfs -put 命令静态导入数据

2.3 删除分区

  • 一次删除多个分区
ALTER TABLE table_name DROP [IF EXISTS] PARTITION (dt='2008-08-08', country='us');

--直接删除数据 不进垃圾桶
ALTER TABLE table_name DROP [IF EXISTS] PARTITION (dt='2008-08-08', country='us') PURGE;

2.4 重命名分区

  • 一次重命名多个分区
ALTER TABLE table_name PARTITION (dt='2008-08-09', country='us') RENAME TO PARTITION (dt='20080809', country='北京');

2.5 修复分区

  • hdfs上的分区与hive元数据中分区字段不一致的情况下,可以使用msck进行修复。
MSCK REPAIR TABLE table_name [ADD/DROP/SYNC PARTITIONS];

-- 比如:直接使用HDFS命令在表文件夹下创建分区文件夹dt='20080810'并上传数据,此时在Hive中
--      查询是无法显示表数据,因为metastore中没有记录,使用MSCK ADD PARTITIONS进行修复。
msck repair table table_name add partitions;

-- 比如:直接使用HDFS命令删除了表文件夹下的分区文件夹dt='20080810',此时在Hive中查询显示
--      分区还在,因为metastore中还没有被删除,使用MSCK DROP PARTITIONS进行修复。
msck repair table table_name drop partitions;

2.6 修改分区

  1. 更改分区文件存储格式
ALTER TABLE table_name PARTITION (dt='2008-08-09') SET FILEFORMAT file_format;
  1. 更改分区位置
ALTER TABLE table_name PARTITION (dt='2008-08-09') SET LOCATION "new location";

3. 分区表的数据导入

一旦创建了分区表,那么在向该分区表中导入数据时,就必须指定分区字段的值,即指定每条数据属于哪一个分区。而:

  1. 如果需要手动指定分区字段的值,叫做静态分区
  2. 如果自动指定分区字段的值,叫动态分区

(1) 静态分区

  1. 语法:使用load data + into table关键字:
load data [local] inpath 'filepath ' 
into table tablename partition(分区字段1='分区值1', 分区字段2='分区值2'...);

直接将文件数据导入到分区表。其实就是将文件导入对应的文件夹下

  1. 例子:
load data local inpath '/root/hivedata/archer.txt' into table t_all_hero_part partition(role='sheshou');
load data local inpath '/root/hivedata/assassin.txt' into table t_all_hero_part partition(role='cike');
load data local inpath '/root/hivedata/mage.txt' into table t_all_hero_part partition(role='fashi');
load data local inpath '/root/hivedata/support.txt' into table t_all_hero_part partition(role='fuzhu');
load data local inpath '/root/hivedata/tank.txt' into table t_all_hero_part partition(role='tanke');
load data local inpath '/root/hivedata/warrior.txt' into table t_all_hero_part partition(role='zhanshi');

然后,在hdfs中查看可知,文件结构由表名文件夹/文件名变为表名文件夹/分区字段=分区值/文件名:中间多了一层文件夹。

hive sql partition 一次插入多分区 hive create table as select 分区_字段

对比创建普通表,不是分区表时,hdfs中文件结构为:

hive sql partition 一次插入多分区 hive create table as select 分区_大数据_02

(2) 动态分区

在静态分区中,可以发现有几个分区值,就要使用load语法分几次导入数据,非常麻烦。

动态分区解决了这个问题,只需要一次加载全部数据,就可以为全部数据打上标签。

  1. 首先要预先设置动态分区:
-- 是否开启动态分区功能。如果该表在创建时有partition关键字,则这里不需要再写该语句
set hive.exec.dynamic.partition=true;

-- 指定动态分区模式,分为nonstick非严格模式和strict严格模式。
-- strict严格模式要求至少有一个分区为静态分区。
set hive.exec.dynamic.partition.mode=nonstrict;
  1. 语法:使用insert + select 语法导入数据
insert into table 分区表名称 partition(分区字段1名称, 分区字段2名称) 
select 
	...,
	分区字段1按照table_name的哪个字段值分区,
	分区字段2按照table_name的哪个字段值分区
from table_name;

将表的查询结果导入到分区表

  1. 例子:
--创建一张新的分区表 t_all_hero_part_dynamic
create table t_all_hero_part_dynamic(
	id int,
	name string,
	hp_max int,
	mp_max int,
	attack_max int,
	defense_max int,
	attack_range string,
	role_main string,
	role_assist string
) partitioned by (role string)
row format delimited fields terminated by "\t";

--执行动态分区插入
insert into table t_all_hero_part_dynamic partition(role) 
select 
	tmp.*,
	tmp.role_main
from t_all_hero tmp;

4. 查询

4.1 查询分区表数据

对于分区表的查询,尽量先使用where进行分区过滤,查询指定分区的数据,避免全表扫描。

比如:

-- role是分区字段
select count(*) from t_all_hero_part where role="sheshou" and hp_max >6000;

4.2 查询分区表结构

  1. 查看分区表有哪些分区字段
desc formatted table_name;

hive sql partition 一次插入多分区 hive create table as select 分区_字段_03

  1. 查看分区表分了几个区
show partitions table_name;

hive sql partition 一次插入多分区 hive create table as select 分区_hive_04

5. 小结

  • 分区表好处:查询时可以避免全表扫描,提高查询效率。

hive sql partition 一次插入多分区 hive create table as select 分区_mysql_05