복잡한 유형으로 Spark SQL DataFrame 쿼리
맵 / 배열과 같은 복잡한 유형으로 RDD를 쿼리하려면 어떻게해야합니까? 예를 들어,이 테스트 코드를 작성할 때 :
case class Test(name: String, map: Map[String, String])
val map = Map("hello" -> "world", "hey" -> "there")
val map2 = Map("hello" -> "people", "hey" -> "you")
val rdd = sc.parallelize(Array(Test("first", map), Test("second", map2)))
구문은 다음과 같습니다.
sqlContext.sql("SELECT * FROM rdd WHERE map.hello = world")
또는
sqlContext.sql("SELECT * FROM rdd WHERE map[hello] = world")
그러나 나는 얻는다
MapType (StringType, StringType, true) 유형의 중첩 필드에 액세스 할 수 없습니다.
과
org.apache.spark.sql.catalyst.errors.package $ TreeNodeException : 해결되지 않은 속성
각기.
컬럼 유형에 따라 다릅니다. 더미 데이터부터 시작하겠습니다.
import org.apache.spark.sql.functions.{udf, lit}
import scala.util.Try
case class SubRecord(x: Int)
case class ArrayElement(foo: String, bar: Int, vals: Array[Double])
case class Record(
an_array: Array[Int], a_map: Map[String, String],
a_struct: SubRecord, an_array_of_structs: Array[ArrayElement])
val df = sc.parallelize(Seq(
Record(Array(1, 2, 3), Map("foo" -> "bar"), SubRecord(1),
Array(
ArrayElement("foo", 1, Array(1.0, 2.0, 2.0)),
ArrayElement("bar", 2, Array(3.0, 4.0, 5.0)))),
Record(Array(4, 5, 6), Map("foz" -> "baz"), SubRecord(2),
Array(ArrayElement("foz", 3, Array(5.0, 6.0)),
ArrayElement("baz", 4, Array(7.0, 8.0))))
)).toDF
df.registerTempTable("df")
df.printSchema
// root
// |-- an_array: array (nullable = true)
// | |-- element: integer (containsNull = false)
// |-- a_map: map (nullable = true)
// | |-- key: string
// | |-- value: string (valueContainsNull = true)
// |-- a_struct: struct (nullable = true)
// | |-- x: integer (nullable = false)
// |-- an_array_of_structs: array (nullable = true)
// | |-- element: struct (containsNull = true)
// | | |-- foo: string (nullable = true)
// | | |-- bar: integer (nullable = false)
// | | |-- vals: array (nullable = true)
// | | | |-- element: double (containsNull = false)
배열 (
ArrayType
) 열 :Column.getItem
방법df.select($"an_array".getItem(1)).show // +-----------+ // |an_array[1]| // +-----------+ // | 2| // | 5| // +-----------+
하이브 대괄호 구문 :
sqlContext.sql("SELECT an_array[1] FROM df").show // +---+ // |_c0| // +---+ // | 2| // | 5| // +---+
UDF
val get_ith = udf((xs: Seq[Int], i: Int) => Try(xs(i)).toOption) df.select(get_ith($"an_array", lit(1))).show // +---------------+ // |UDF(an_array,1)| // +---------------+ // | 2| // | 5| // +---------------+
위에 나열된 방법 외에도 Spark는 복잡한 유형에서 작동하는 내장 함수의 증가하는 목록을 지원합니다. 주목할만한 예에는
transform
(SQL 전용, 2.4+) 와 같은 고차 함수가 포함됩니다 .df.selectExpr("transform(an_array, x -> x + 1) an_array_inc").show // +------------+ // |an_array_inc| // +------------+ // | [2, 3, 4]| // | [5, 6, 7]| // +------------+
filter
(SQL 전용, 2.4 이상)df.selectExpr("filter(an_array, x -> x % 2 == 0) an_array_even").show // +-------------+ // |an_array_even| // +-------------+ // | [2]| // | [4, 6]| // +-------------+
aggregate
(SQL 전용, 2.4 이상) :df.selectExpr("aggregate(an_array, 0, (acc, x) -> acc + x, acc -> acc) an_array_sum").show // +------------+ // |an_array_sum| // +------------+ // | 6| // | 15| // +------------+
(2.4+)
array_*
와 같은 배열 처리 함수 (array_distinct
) :import org.apache.spark.sql.functions.array_distinct df.select(array_distinct($"an_array_of_structs.vals"(0))).show // +-------------------------------------------+ // |array_distinct(an_array_of_structs.vals[0])| // +-------------------------------------------+ // | [1.0, 2.0]| // | [5.0, 6.0]| // +-------------------------------------------+
array_max
(array_min
, 2.4 이상) :import org.apache.spark.sql.functions.array_max df.select(array_max($"an_array")).show // +-------------------+ // |array_max(an_array)| // +-------------------+ // | 3| // | 6| // +-------------------+
flatten
(2.4 이상)import org.apache.spark.sql.functions.flatten df.select(flatten($"an_array_of_structs.vals")).show // +---------------------------------+ // |flatten(an_array_of_structs.vals)| // +---------------------------------+ // | [1.0, 2.0, 2.0, 3...| // | [5.0, 6.0, 7.0, 8.0]| // +---------------------------------+
arrays_zip
(2.4+) :import org.apache.spark.sql.functions.arrays_zip df.select(arrays_zip($"an_array_of_structs.vals"(0), $"an_array_of_structs.vals"(1))).show(false) // +--------------------------------------------------------------------+ // |arrays_zip(an_array_of_structs.vals[0], an_array_of_structs.vals[1])| // +--------------------------------------------------------------------+ // |[[1.0, 3.0], [2.0, 4.0], [2.0, 5.0]] | // |[[5.0, 7.0], [6.0, 8.0]] | // +--------------------------------------------------------------------+
array_union
(2.4+) :import org.apache.spark.sql.functions.array_union df.select(array_union($"an_array_of_structs.vals"(0), $"an_array_of_structs.vals"(1))).show // +---------------------------------------------------------------------+ // |array_union(an_array_of_structs.vals[0], an_array_of_structs.vals[1])| // +---------------------------------------------------------------------+ // | [1.0, 2.0, 3.0, 4...| // | [5.0, 6.0, 7.0, 8.0]| // +---------------------------------------------------------------------+
slice
(2.4+) :import org.apache.spark.sql.functions.slice df.select(slice($"an_array", 2, 2)).show // +---------------------+ // |slice(an_array, 2, 2)| // +---------------------+ // | [2, 3]| // | [5, 6]| // +---------------------+
지도 (
MapType
) 열사용
Column.getField
방법 :df.select($"a_map".getField("foo")).show // +----------+ // |a_map[foo]| // +----------+ // | bar| // | null| // +----------+
Hive 대괄호 구문 사용 :
sqlContext.sql("SELECT a_map['foz'] FROM df").show // +----+ // | _c0| // +----+ // |null| // | baz| // +----+
점 구문으로 전체 경로 사용 :
df.select($"a_map.foo").show // +----+ // | foo| // +----+ // | bar| // |null| // +----+
UDF 사용
val get_field = udf((kvs: Map[String, String], k: String) => kvs.get(k)) df.select(get_field($"a_map", lit("foo"))).show // +--------------+ // |UDF(a_map,foo)| // +--------------+ // | bar| // | null| // +--------------+
의 증가
map_*
같은 기능을 수행map_keys
(2.3)import org.apache.spark.sql.functions.map_keys df.select(map_keys($"a_map")).show // +---------------+ // |map_keys(a_map)| // +---------------+ // | [foo]| // | [foz]| // +---------------+
또는
map_values
(2.3+)import org.apache.spark.sql.functions.map_values df.select(map_values($"a_map")).show // +-----------------+ // |map_values(a_map)| // +-----------------+ // | [bar]| // | [baz]| // +-----------------+
자세한 목록은 SPARK-23899 를 확인 하십시오.
StructType
점 구문으로 전체 경로를 사용하는 struct ( ) 열 :DataFrame API 사용
df.select($"a_struct.x").show // +---+ // | x| // +---+ // | 1| // | 2| // +---+
원시 SQL 사용
sqlContext.sql("SELECT a_struct.x FROM df").show // +---+ // | x| // +---+ // | 1| // | 2| // +---+
의 배열 내부 필드
structs
는 점 구문, 이름 및 표준Column
방법을 사용하여 액세스 할 수 있습니다 .df.select($"an_array_of_structs.foo").show // +----------+ // | foo| // +----------+ // |[foo, bar]| // |[foz, baz]| // +----------+ sqlContext.sql("SELECT an_array_of_structs[0].foo FROM df").show // +---+ // |_c0| // +---+ // |foo| // |foz| // +---+ df.select($"an_array_of_structs.vals".getItem(1).getItem(1)).show // +------------------------------+ // |an_array_of_structs.vals[1][1]| // +------------------------------+ // | 4.0| // | 8.0| // +------------------------------+
사용자 정의 유형 (UDT) 필드는 UDF를 사용하여 액세스 할 수 있습니다. 자세한 내용 은 UDT의 SparkSQL 참조 속성을 참조 하세요.
참고 :
- depending on a Spark version some of these methods can be available only with
HiveContext
. UDFs should work independent of version with both standardSQLContext
andHiveContext
. generally speaking nested values are a second class citizens. Not all typical operations are supported on nested fields. Depending on a context it could be better to flatten the schema and / or explode collections
df.select(explode($"an_array_of_structs")).show // +--------------------+ // | col| // +--------------------+ // |[foo,1,WrappedArr...| // |[bar,2,WrappedArr...| // |[foz,3,WrappedArr...| // |[baz,4,WrappedArr...| // +--------------------+
Dot syntax can be combined with wildcard character (
*
) to select (possibly multiple) fields without specifying names explicitly:df.select($"a_struct.*").show // +---+ // | x| // +---+ // | 1| // | 2| // +---+
JSON columns can be queried using
get_json_object
andfrom_json
functions. See How to query JSON data column using Spark DataFrames? for details.
Once You convert it to DF, u can simply fetch data as
val rddRow= rdd.map(kv=>{
val k = kv._1
val v = kv._2
Row(k, v)
})
val myFld1 = StructField("name", org.apache.spark.sql.types.StringType, true)
val myFld2 = StructField("map", org.apache.spark.sql.types.MapType(StringType, StringType), true)
val arr = Array( myFld1, myFld2)
val schema = StructType( arr )
val rowrddDF = sqc.createDataFrame(rddRow, schema)
rowrddDF.registerTempTable("rowtbl")
val rowrddDFFinal = rowrddDF.select(rowrddDF("map.one"))
or
val rowrddDFFinal = rowrddDF.select("map.one")
here was what I did and it worked
case class Test(name: String, m: Map[String, String])
val map = Map("hello" -> "world", "hey" -> "there")
val map2 = Map("hello" -> "people", "hey" -> "you")
val rdd = sc.parallelize(Array(Test("first", map), Test("second", map2)))
val rdddf = rdd.toDF
rdddf.registerTempTable("mytable")
sqlContext.sql("select m.hello from mytable").show
Results
+------+
| hello|
+------+
| world|
|people|
+------+
ReferenceURL : https://stackoverflow.com/questions/28332494/querying-spark-sql-dataframe-with-complex-types
'programing' 카테고리의 다른 글
재생하기 전에 videoview에서 미리보기 이미지를 설정하는 방법 (0) | 2021.01.17 |
---|---|
CSS에서 배경 이미지 경로를 제공하는 방법은 무엇입니까? (0) | 2021.01.17 |
개조 컨버터 공장에서 GsonConverterFactory에 액세스 할 수 없습니다. (0) | 2021.01.17 |
wpf를 사용하여 dll을 단일 .exe로 병합 (0) | 2021.01.17 |
ECMAScript v 6이 표준이되는시기 (0) | 2021.01.16 |