zoukankan      html  css  js  c++  java
  • Spark学习之Spark SQL

    一、简介

      Spark SQL 提供了以下三大功能。
       (1) Spark SQL 可以从各种结构化数据源(例如 JSON、Hive、Parquet 等)中读取数据。
       (2) Spark SQL 不仅支持在 Spark 程序内使用 SQL 语句进行数据查询,也支持从类似商业智能软件 Tableau 这样的外部工具中通过标准数据库连接器(JDBC/ODBC)连接 SparkSQL 进行查询。
       (3) 当在 Spark 程序内使用 Spark SQL 时,Spark SQL 支持 SQL 与常规的 Python/Java/Scala代码高度整合,包括连接 RDD 与 SQL 表、公开的自定义 SQL 函数接口等。这样一来,许多工作都更容易实现了。

    二、Spark SQL基本示例

      

    import org.apache.spark.SparkContext
    import org.apache.spark.SparkConf
    import org.apache.spark.sql.hive.HiveContext
    import org.apache.spark.sql.SQLContext
    
    object Test {
      def main(args: Array[String]): Unit = {
        
        val conf = new SparkConf().setAppName("test").setMaster("local[4]")
        val sc = new SparkContext(conf)
        sc.setLogLevel("WARN")  // 设置日志显示级别
        val hiveCtx = new HiveContext(sc)
        val input = hiveCtx.jsonFile("tweets.json")
        input.registerTempTable("tweets")
        // hiveCtx.cacheTable("tweets") 缓存表
        // input.printSchema()  // 输出结构信息
        val topTweets = hiveCtx.sql("SELECT user.name,text FROM tweets")
        topTweets.collect().foreach(println)
        
      }
    }

      

    三、SchemaRDD

      读取数据和执行查询都会返回 SchemaRDD。SchemaRDD 和传统数据库中的表的概念类似。从内部机理来看,SchemaRDD 是一个由 Row 对象组成的 RDD,附带包含每列数据类型的结构信息。 Row 对象只是对基本数据类型(如整型和字符串型等)的数组的封装。在 Spark 1.3 版本以后,SchemaRDD 这个名字被改为 DataFrame。SchemaRDD 仍然是 RDD,所以你可以对其应用已有的 RDD 转化操作,比如 map() 和filter() 。

      SchemaRDD 可以存储一些基本数据类型,也可以存储由这些类型组成的结构体和数组。

      

        

    四、缓存

      Spark SQL 的缓存机制与 Spark 中的稍有不同。由于我们知道每个列的类型信息,所以Spark 可以更加高效地存储数据。为了确保使用更节约内存的表示方式进行缓存而不是储存整个对象,应当使用专门的 hiveCtx.cacheTable("tableName") 方法。当缓存数据表时,Spark SQL 使用一种列式存储格式在内存中表示数据。这些缓存下来的表只会在驱动器程序的生命周期里保留在内存中,所以如果驱动器进程退出,就需要重新缓存数据。和缓存RDD 时的动机一样,如果想在同样的数据上多次运行任务或查询时,就应把这些数据表缓存起来。

    五、读取和存储数据

    1、 Apache Hive  

      当从 Hive 中读取数据时,Spark SQL 支持任何 Hive 支持的存储格式,包括文本文件、RCFiles、ORC、Parquet、Avro,以及 Protocol Buffer。要把 Spark SQL 连接到已经部署好的 Hive 上,你需要提供一份 Hive 配置。你只需要把你的 hive-site.xml 文件复制到 Spark 的 ./conf/ 目录下即可。如果你只是想探索一下 Spark SQL而没有配置 hive-site.xml 文件,那么 Spark SQL 则会使用本地的 Hive 元数据仓,并且同样可以轻松地将数据读取到 Hive 表中进行查询。

    // 使用 Scala 从 Hive 读取
    val hiveCtx = new HiveContext(sc)
    val rows = hiveCtx.sql("SELECT key, value FROM mytable")
    val keys = rows.map(row => row.getInt(0))
    

    2、Parquet

      Parquet(http://parquet.apache.org/)是一种流行的列式存储格式,可以高效地存储具有嵌套字段的记录。Parquet 格式经常在 Hadoop 生态圈中被使用,它也支持 Spark SQL 的全部数据类型。Spark SQL 提供了直接读取和存储 Parquet 格式文件的方法。可以通过 HiveContext.parquetFile 或者 SQLContext.parquetFile 来读取数据。

    3、JSON

      要读取 JSON 数据,只要调用 hiveCtx 中的 jsonFile() 方法即可。如果你想获得从数据中推断出来的结构信息,可以在生成的 SchemaRDD 上调用printSchema 方法。

    4、基于RDD

      除了读取数据,也可以基于 RDD 创建 DataFrame。

    /**SQLContext 创建 DataFrame **/
      def createDtaFrame(sparkCtx:SparkContext,sqlCtx:SQLContext):Unit = {
        val rowRDD = sparkCtx.textFile("D://TxtData/studentInfo.txt").map(_.split(",")).map(p => Row(p(0),p(1).toInt,p(2)))
        val schema = StructType(
          Seq(
            StructField("name",StringType,true),
            StructField("age",IntegerType,true),
            StructField("studentNo",StringType,true)
          )
        )
        val dataDF = sqlCtx.createDataFrame(rowRDD,schema)
     
        //df注册到内存表
        dataDF.registerTempTable("Student")
        val result = sqlCtx.sql("select * from Student")
        result.show()
     
        //    dataDF.select("name").show()
        //    dataDF.filter(dataDF("age") <14).show()
        //    dataDF.where("age <> ''").show()
      }
    

    5、Spark SQL UDF

      用户自定义函数,也叫 UDF,可以让我们使用 Python/Java/Scala 注册自定义函数,并在 SQL中调用。这种方法很常用,通常用来给机构内的 SQL 用户们提供高级功能支持,这样这些用户就可以直接调用注册的函数而无需自己去通过编程来实现了。在 Spark SQL 中,编写UDF 尤为简单。Spark SQL 不仅有自己的 UDF 接口,也支持已有的 Apache Hive UDF。

        //  写一个求字符串长度的UDF  原书代码会报错,查阅官方文档得知  
        hiveCtx.udf.register("strLenScala", (_: String).length)
        val tweetLength = hiveCtx.sql("SELECT strLenScala('tweet') FROM tweets LIMIT 10")
    

      注意:因为Spark版本的原因,原书的代码会报错,有一个好的解决方式,就是查阅官方文档。

      

    六、Spark SQL性能

      Spark SQL 提供的高级查询语言及附加的类型信息可以使 SparkSQL 数据查询更加高效。

      Spark SQL 的性能调优选项如下表所示

      

      在一个传统的 Spark SQL 应用中,可以在 Spark 配置中设置这些 Spark 属性。

    // 在 Scala 中打开 codegen 选项的代码
    conf.set("spark.sql.codegen", "true")

      Spark Sql官方文档

      这篇博文主要来自《Spark快速大数据分析》这本书里面的第九章,内容有删减,还有本书的一些代码的实验结果。还要注意一点,在以后的学习中要养成查阅官方文档的习惯。

  • 相关阅读:
    Java虚拟机工作原理具体解释
    关于java的JIT知识
    php重建二叉树(函数缺省参数相关的都写在后面,比如array_slice函数中的$length属性,故第一个参数是操作的数组)
    php实现从尾到头打印列表
    thinkphp5项目--企业单车网站(九)(加强复习啊)(花了那么多时间写的博客,不复习太浪费了)
    php 面试题一(看视频的学习量比网上瞎转悠要清晰和明了很多)(看视频做好笔记)(注重复习)
    php对象和数组的相互转换(还是可以去找没有没php的高阶课程看看看)(要不别人分析一下重点要点,要不自己来,不然 效果真的不好)
    js中JSON的解析(将json字符串转化为对象)和序列化(将对象转化为json字符串)(函数的功能一般都挺全的,需要的时候去查看完整函数)
    html中radio、checkbox选中状态研究(静下心来看,静下心来总结)
    thinkphp中view页面中的volist标签转化为原生php分析(多去看源代码,你会发现不仅简单,方便你理解,还节约时间)
  • 原文地址:https://www.cnblogs.com/xiaoyh/p/10778975.html
Copyright © 2011-2022 走看看