Hive源码分析:CLI入口类

简介:

说明:

本文的源码分析基于hive-0.10.0-cdh4.3.0。

启动脚本

从shell脚本/usr/lib/hive/bin/ext/cli.sh可以看到hive cli的入口类为org.apache.hadoop.hive.cli.CliDriver

	cli () {
	  CLASS=org.apache.hadoop.hive.cli.CliDriver
	  execHiveCmd $CLASS "$@"
	}
	cli_help () {
	  CLASS=org.apache.hadoop.hive.cli.CliDriver
	  execHiveCmd $CLASS "--help"
	}

入口类

java中的类如果有main方法就能运行,故直接查找org.apache.hadoop.hive.cli.CliDriver中的main方法即可。

CliDriver类中的方法有:

main方法代码如下:

	public static void main(String[] args) throws Exception {
	    int ret = run(args);
	    System.exit(ret);
	}

阅读run函数可以看到,主要做了以下几件事情:

  • 读取main方法的参数
  • 重置默认的log4j配置并为hive重新初始化log4j,注意,在这里是读取hive-log4j.properties来初始化log4j。
  • 创建CliSessionState,并初始化in、out、info、error等stream流。CliSessionState是一次命令行操作的session会话,其继承了SessionState。
  • 重命令行参数中读取参数并设置到CliSessionState中。
  • 启动SessionState并连接到hive server
  • 如果cli是本地模式运行,则加载hive.aux.jars.path参数配置的jar包到classpath
  • 创建一个CliDriver对象,并设置当前选择的数据库。可以在命令行参数添加-database database来选择连接那个数据库,默认为default数据库。
  • 加载初始化文件.hiverc,该文件位于当前用户主目录下,读取该文件内容后,然后调用processFile方法处理文件内容。
  • 如果命令行中有-e参数,则运行指定的sql语句;如果有-f参数,则读取该文件内容并运行。注意:不能同时指定这两个参数。
	hive -e 'show tables'
	hive -f /root/hive.sql
  • 如果没有指定上面两个参数,则从当前用户主目录读取.hivehistory文件,如果不存在则创建。该文件保存了当前用户所有运行的hive命令。
  • 在while循环里不断读取控制台的输入内容,每次读取一行,如果行末有分号,则调用CliDriver的processLine方法运行读取到的内容。
  • 每次调用processLine方法时,都会创建SignalHandler用于捕捉用户的输入,当用户输入Ctrl+C时,会kill当前正在运行的任务以及kill掉当前进程。kill当前正在运行的job的代码如下.
	HadoopJobExecHelper.killRunningJobs();
  • 处理hive命令。

处理hive命令过程

如果输入的是quit或者exit,则程序退出。

如果命令开头是source,则会读取source 后面文件内容,然后执行该文件内容。通过这种方式,你可以在hive命令行模式运行一个文件中的hive命令。

如果命令开头是感叹号,执行操作系统命令(如!ls,列出当前目录的文件信息)。通过以下代码来运行:

	Process executor = Runtime.getRuntime().exec(shell_cmd);
	StreamPrinter outPrinter = new StreamPrinter(executor.getInputStream(), null, ss.out);
	StreamPrinter errPrinter = new StreamPrinter(executor.getErrorStream(), null, ss.err);

	outPrinter.start();
	errPrinter.start();

	ret = executor.waitFor();
	if (ret != 0) {
	  console.printError("Command failed with exit code = " + ret);
	}

shell_cmd的内容大概如下:

	shell_cmd = "/bin/bash -c \'" + shell_cmd + "\'"

如果命令开头是list,列出jar/file/archive

如果是远程模式运行命令行,则通过HiveClient来运行命令;否则,调用processLocalCmd方法运行本地命令。

以本地模式运行时,会通过CommandProcessorFactory工厂解析输入的语句来获得一个CommandProcessor,CommandProcessor接口的实现类见下图:

从上图可以看到指定的命令(set/dfs/add/delete/reset)交给指定的CommandProcessor处理,其余的(指hql语句)交给Driver类来处理。

故,org.apache.hadoop.hive.ql.Driver类是hql查询的起点,而run()方法会先后调用compile()和execute()两个函数来完成查询,所以一个command的查询分为compile和execute两个阶段。

总结

作为尝试,第一次使用思维导图分析代码逻辑,简单整理了一下CliDriver类的运行逻辑,如下图。以后还需要加强画图和表达能力。

hive-cli-clidriver

目录
相关文章
|
8月前
|
SQL 分布式计算 HIVE
Hive Cli / HiveServer2 中使用 dayofweek 函数引发的BUG!
在Hive 3.1.2和Spark 3.0.2集群环境中,遇到`dayofweek`函数bug。当`create_date`为字符串类型时,`dayofweek`函数结果错位。修复方法是将`create_date`转换为`date`类型。在Spark SQL中,原始代码能正常运行,未出现此问题。因此建议在Hive中使用转换后的日期类型以避免错误。
96 4
|
8月前
|
SQL 存储 分布式计算
Hive详解、配置、数据结构、Hive CLI
Hive详解、配置、数据结构、Hive CLI
137 0
Hive详解、配置、数据结构、Hive CLI
|
8月前
|
SQL 存储 分布式计算
Hive【基础知识 02-2】【Hive CLI 命令行工具使用】【详细举例-包含测试脚本文件】
【4月更文挑战第7天】Hive【基础知识 02-2】【Hive CLI 命令行工具使用】【详细举例-包含测试脚本文件】
126 0
|
8月前
|
SQL 关系型数据库 MySQL
Hive【基础知识 02-1】【Hive CLI 命令行工具使用】【准备阶段-建库、建表、导入数据、编写测试SQL脚本并上传HDFS】
【4月更文挑战第7天】Hive【基础知识 02-1】【Hive CLI 命令行工具使用】【准备阶段-建库、建表、导入数据、编写测试SQL脚本并上传HDFS】
95 0
|
SQL 数据库 HIVE
数据仓库的Hive的Hive架构的用户接口的CLI
Hive提供了一系列用户接口,可以方便地进行数据仓库的管理和操作。其中,CLI(命令行界面)是一种非常常用的用户接口,可以通过命令行进行Hive的管理和操作。
159 0
|
SQL 分布式计算 Hadoop
|
SQL 分布式计算 Kubernetes
spark hive类总是优先记载应用里面的jar包,跟spark.{driver/executor}.userClassPathFirst无关
spark hive类总是优先记载应用里面的jar包,跟spark.{driver/executor}.userClassPathFirst无关
301 0
|
SQL JSON Oracle
hive函数大全:11大类、109个函数(下)
hive函数大全:11大类、109个函数
hive函数大全:11大类、109个函数(上)
hive函数大全:11大类、109个函数(上)
|
SQL 大数据 Java
hive(在大数据集合上的类SQL查询和表)学习
hive(在大数据集合上的类SQL查询和表)学习
162 0