1、什么是HiveContext
  • Spark SQL支持对Hive中存储的数据进行读写
    操作Hive中的数据时,必须创建HiveContext(HiveContext也是已经过时的不推荐使用,嘤嘤嘤~还没学好就已经过时了)。HiveContext继承自SQLContext,但是增加了在Hive元数据库中查找表,以及HiveSQL语法编写SQL的功能。除了sql()方法,HiveContext还提供了hql()方法,从而用Hive语法来编辑sql。
  • Spark SQL允许将数据保存到Hive表中
    调用DataFrame的saveAsTable命令,即可将DataFrame中的数据保存到Hive表中。与registerTempTable不同,saveAsTable是会将DataFrame中的数据物化到Hive表中的,而且还会在Hive元数据库中创建表的元数据。
    默认情况下,saveAsTable会创建一张Hive Managed Table,也就是说数据的位置都是由元数据库中的信息控制的。当Managed Table被删除时,表中的数据也会一并被物理剔除。
    registerTempTable只是注册一个临时的表,只要Spark Application重启或者停止了,那么表就没了。而saveAsTable创建的是物化的表,无论Spark Application重启或者停止,表都会一直存在。
    调用HiveContext.table()方法,还可以直接针对Hive中的表,创建一个DataFrame。
2、如何使用HiveContext

使用HiveContext,可以执行Hive的大部分功能,包括创建表、向表中导入数据以及用SQL语句查询表中的数据。查询出来的数据是一个Row数组。

将hive-site.xml拷贝到spark/conf目录下,将mysql connector拷贝到spark/lib目录下

dophin scheduler Hive数据源 hive source sql_sql


dophin scheduler Hive数据源 hive source sql_spark_02

HiveContext sqlContext = new HiveContext(sc);
sqlContext.sql("CREATE TABLE IF NOT EXISTS students (name STRING,age INT)");
sqlContext.sql("LOAD DATA LOCAL INPATH '/usr/local/spark-study/resources/students.txt' INTO TABLE students");
Row[] teenagers = sqlContext.sql("SELECT name,age FROM students WHERE age<=18").collect();
3、案例:查询分数大于80分的学生的信息
  • java版
package pz.spark.study.sql;

import org.apache.spark.SparkConf;
import org.apache.spark.api.java.JavaSparkContext;
import org.apache.spark.sql.Dataset;
import org.apache.spark.sql.Row;
import org.apache.spark.sql.hive.HiveContext;

public class HiveDataSource {
    public static void main(String[] args) {
        //创建SparkConf
        SparkConf conf = new SparkConf().setAppName("HiveDataSource");
        //创建JavaSparkContext
        JavaSparkContext sc = new JavaSparkContext(conf);
        //创建hiveContext,注意这里接收的是SparkContext作为参数,而不是JavaSparkContext
        HiveContext hiveContext = new HiveContext(sc.sc());

        //第一个功能,使用HiveContext的sql()/hql()方法,可以执行hive中能够执行的HiveSQL语句
        //将学生基本信息数据导入student_infos表
        //判断是否存在student_infos,如果存在则删除
        hiveContext.sql("DROP TABLE IF EXISTS TABLE student_infos");
        //判断student_infos表是否不存在,如果不存在则创建该表
        hiveContext.sql("CREATE TABLE IF NOT EXISTS student_infos (name STRING,age INT)");

        //将学生基本信息数据导入student_infos表
        hiveContext.sql("LOAD DATA LOCAL INPATH '/usr/local/spark-study/student_infos.txt' INTO TABLE student_infos");

        //用同样的方式给student_scores导入数据
        hiveContext.sql("DROP TABLE IF EXISTS TABLE student_scores");
        hiveContext.sql("CREATE TABLE IF NOT EXISTS  student_scores (name STRING,score INT)");
        hiveContext.sql("LOAD DATA LOCAL INPATH '/usr/local/spark-study/student_scores.txt' INTO TABLE student_scores");

        //第二个功能,执行sql还可以返回DataFrame,用于查询
        //执行sql查询,关联两张表,查询成绩大于80分的学生信息
        Dataset<Row> goodStrudentsDF = hiveContext.sql("SELECT si.name name,si.age age,ss.score" +
                "FROM student_infos si" +
                "JOIN studnet_scores ss " +
                "ON si.name=ss.name" +
                "WHERE ss.score>=80");

        //第三个功能,可以将DataFrame中的数据,理论上说DataFrame对应的RDD的元素,是Row即可

        //将DataFrame中的数据保存到good_student_infos表中
        hiveContext.sql("DROP IF EXISTS TABLE good_student_infos");
        goodStrudentsDF.write().saveAsTable("good_student_infos");

        //第四个工程,可以用table()方法,针对hive表,直接创建DataFrame
        //然后针对good_student_infos表,直接创建DataFrame
        Row[] goodStudentRows = hiveContext.table("good_student_infos").collect();

        for (Row goodStudentRow :goodStudentRows) {
            System.out.println(goodStudentRow);
        }
        sc.close();
    }
}
  • scala版
package pz.spark.study.sql

import org.apache.spark.SparkConf
import org.apache.spark.api.java.JavaSparkContext
import org.apache.spark.sql.hive.HiveContext

object HiveDataSource_scala {
  def main(args: Array[String]): Unit = {
    //创建SparkConf
    val conf = new SparkConf().setAppName("HiveDataSource")
    //创建JavaSparkContext
    val sc = new JavaSparkContext(conf)
    //创建hiveContext,注意这里接收的是SparkContext作为参数,而不是JavaSparkContext
    val hiveContext = new HiveContext(sc)

    //第一个功能,使用HiveContext的sql()/hql()方法,可以执行hive中能够执行的HiveSQL语句
    //将学生基本信息数据导入student_infos表
    //判断是否存在student_infos,如果存在则删除
    hiveContext.sql("DROP TABLE IF EXISTS TABLE student_infos")
    //判断student_infos表是否不存在,如果不存在则创建该表
    hiveContext.sql("CREATE TABLE IF NOT EXISTS student_infos (name STRING,age INT)")


    hiveContext.sql("LOAD DATA LOCAL INPATH '/usr/local/spark-study/student_infos.txt' INTO TABLE student_infos")

    //用同样的方式给student_scores导入数据
    hiveContext.sql("DROP TABLE IF EXISTS TABLE student_scores")
    hiveContext.sql("CREATE TABLE IF NOT EXISTS  student_scores (name STRING,score INT)")
    hiveContext.sql("LOAD DATA LOCAL INPATH '/usr/local/spark-study/student_scores.txt' INTO TABLE student_scores")

    //第二个功能,执行sql还可以返回DataFrame,用于查询
    //执行sql查询,关联两张表,查询成绩大于80分的学生信息
    val goodStrudentsDF = hiveContext.sql("SELECT si.name name,si.age age,ss.score" + "FROM student_infos si" + "JOIN studnet_scores ss " + "ON si.name=ss.name" + "WHERE ss.score>=80")

    //第三个功能,可以将DataFrame中的数据,理论上说DataFrame对应的RDD的元素,是Row即可

    //将DataFrame中的数据保存到good_student_infos表中
    hiveContext.sql("DROP IF EXISTS TABLE good_student_infos")
    goodStrudentsDF.write.saveAsTable("good_student_infos")

    //第四个工程,可以用table()方法,针对hive表,直接创建DataFrame
    //然后针对good_student_infos表,直接创建DataFrame
    val goodStudentRows = hiveContext.table("good_student_infos").collect

    for (goodStudentRow <- goodStudentRows) {
      System.out.println(goodStudentRow)
    }
    sc.close()
  }
}