Искра> = 2.3.0
SPARK-22614 предоставляет разделение диапазона.
val partitionedByRange = df.repartitionByRange(42, $"k")
partitionedByRange.explain
// == Parsed Logical Plan ==
// 'RepartitionByExpression ['k ASC NULLS FIRST], 42
// +- AnalysisBarrier Project [_1#2 AS k#5, _2#3 AS v#6]
//
// == Analyzed Logical Plan ==
// k: string, v: int
// RepartitionByExpression [k#5 ASC NULLS FIRST], 42
// +- Project [_1#2 AS k#5, _2#3 AS v#6]
// +- LocalRelation [_1#2, _2#3]
//
// == Optimized Logical Plan ==
// RepartitionByExpression [k#5 ASC NULLS FIRST], 42
// +- LocalRelation [k#5, v#6]
//
// == Physical Plan ==
// Exchange rangepartitioning(k#5 ASC NULLS FIRST, 42)
// +- LocalTableScan [k#5, v#6]
SPARK-22389 предоставляет разделение по внешнему формату в Data Source API v2 .
Искра> = 1.6.0
В Spark> = 1.6 можно использовать разбиение по столбцам для запросов и кеширования. См: SPARK-11410 и СПАРК-4849 с использованием repartition
метода:
val df = Seq(
("A", 1), ("B", 2), ("A", 3), ("C", 1)
).toDF("k", "v")
val partitioned = df.repartition($"k")
partitioned.explain
// scala> df.repartition($"k").explain(true)
// == Parsed Logical Plan ==
// 'RepartitionByExpression ['k], None
// +- Project [_1#5 AS k#7,_2#6 AS v#8]
// +- LogicalRDD [_1#5,_2#6], MapPartitionsRDD[3] at rddToDataFrameHolder at <console>:27
//
// == Analyzed Logical Plan ==
// k: string, v: int
// RepartitionByExpression [k#7], None
// +- Project [_1#5 AS k#7,_2#6 AS v#8]
// +- LogicalRDD [_1#5,_2#6], MapPartitionsRDD[3] at rddToDataFrameHolder at <console>:27
//
// == Optimized Logical Plan ==
// RepartitionByExpression [k#7], None
// +- Project [_1#5 AS k#7,_2#6 AS v#8]
// +- LogicalRDD [_1#5,_2#6], MapPartitionsRDD[3] at rddToDataFrameHolder at <console>:27
//
// == Physical Plan ==
// TungstenExchange hashpartitioning(k#7,200), None
// +- Project [_1#5 AS k#7,_2#6 AS v#8]
// +- Scan PhysicalRDD[_1#5,_2#6]
В отличие от RDDs
Spark Dataset
(включая Dataset[Row]
aka DataFrame
), на данный момент нельзя использовать собственный разделитель. Обычно вы можете решить эту проблему, создав столбец искусственного разделения, но это не даст вам такой же гибкости.
Spark <1.6.0:
Одна вещь, которую вы можете сделать, - это предварительно разделить входные данные перед тем, как создавать DataFrame
import org.apache.spark.sql.types._
import org.apache.spark.sql.Row
import org.apache.spark.HashPartitioner
val schema = StructType(Seq(
StructField("x", StringType, false),
StructField("y", LongType, false),
StructField("z", DoubleType, false)
))
val rdd = sc.parallelize(Seq(
Row("foo", 1L, 0.5), Row("bar", 0L, 0.0), Row("??", -1L, 2.0),
Row("foo", -1L, 0.0), Row("??", 3L, 0.6), Row("bar", -3L, 0.99)
))
val partitioner = new HashPartitioner(5)
val partitioned = rdd.map(r => (r.getString(0), r))
.partitionBy(partitioner)
.values
val df = sqlContext.createDataFrame(partitioned, schema)
Поскольку для DataFrame
создания из файла RDD
требуется только простая фаза карты, необходимо сохранить существующий макет раздела *:
assert(df.rdd.partitions == partitioned.partitions)
Таким же образом можно переделать существующие DataFrame
:
sqlContext.createDataFrame(
df.rdd.map(r => (r.getInt(1), r)).partitionBy(partitioner).values,
df.schema
)
Так что, похоже, это не невозможно. Остается вопрос, имеет ли это вообще смысл. Я буду утверждать, что в большинстве случаев это не так:
Переразбиение - дорогостоящий процесс. В типичном сценарии большую часть данных необходимо сериализовать, перемешать и десериализовать. С другой стороны, количество операций, для которых могут быть полезны предварительно разделенные данные, относительно невелико и дополнительно ограничено, если внутренний API не предназначен для использования этого свойства.
- присоединяется в некоторых сценариях, но для этого потребуется внутренняя поддержка,
- вызовы оконных функций с соответствующим разделителем. То же, что и выше, только для определения одного окна. Однако он уже разделен внутри, поэтому предварительное разделение может быть избыточным,
- простое агрегирование с помощью
GROUP BY
- можно уменьшить объем памяти, занимаемой временными буферами **, но общая стоимость намного выше. Более или менее эквивалентно groupByKey.mapValues(_.reduce)
(текущее поведение) vs reduceByKey
(предварительное разбиение). Вряд ли пригодится на практике.
- сжатие данных с помощью
SqlContext.cacheTable
. Поскольку похоже, что используется кодирование длин серий, применение OrderedRDDFunctions.repartitionAndSortWithinPartitions
может улучшить степень сжатия.
Производительность сильно зависит от распределения ключей. Если он перекошен, это приведет к неоптимальному использованию ресурсов. В худшем случае закончить работу будет невозможно.
- Весь смысл использования декларативного API высокого уровня - изолировать себя от деталей реализации низкого уровня. Как уже упоминалось @dwysakowicz и @RomiKuntsman , оптимизация - это работа Catalyst Optimizer . Это довольно сложный зверь, и я действительно сомневаюсь, что вы сможете легко улучшить его, не углубляясь в его внутренности.
Связанные понятия
Разбиение с исходниками JDBC :
predicates
Аргумент поддержки источников данных JDBC . Его можно использовать следующим образом:
sqlContext.read.jdbc(url, table, Array("foo = 1", "foo = 3"), props)
Он создает один раздел JDBC для каждого предиката. Имейте в виду, что если наборы, созданные с использованием отдельных предикатов, не являются непересекающимися, вы увидите дубликаты в итоговой таблице.
partitionBy
метод вDataFrameWriter
:
Spark DataFrameWriter
предоставляет partitionBy
метод, который можно использовать для «разделения» данных при записи. Он разделяет данные при записи, используя предоставленный набор столбцов
val df = Seq(
("foo", 1.0), ("bar", 2.0), ("foo", 1.5), ("bar", 2.6)
).toDF("k", "v")
df.write.partitionBy("k").json("/tmp/foo.json")
Это позволяет предикату выталкивать вниз при чтении для запросов на основе ключа:
val df1 = sqlContext.read.schema(df.schema).json("/tmp/foo.json")
df1.where($"k" === "bar")
но это не эквивалентно DataFrame.repartition
. В частности, такие агрегаты, как:
val cnts = df1.groupBy($"k").sum()
по-прежнему потребуется TungstenExchange
:
cnts.explain
// == Physical Plan ==
// TungstenAggregate(key=[k#90], functions=[(sum(v#91),mode=Final,isDistinct=false)], output=[k#90,sum(v)#93])
// +- TungstenExchange hashpartitioning(k#90,200), None
// +- TungstenAggregate(key=[k#90], functions=[(sum(v#91),mode=Partial,isDistinct=false)], output=[k#90,sum#99])
// +- Scan JSONRelation[k#90,v#91] InputPaths: file:/tmp/foo.json
bucketBy
метод вDataFrameWriter
(Spark> = 2.0):
bucketBy
имеет аналогичные приложения, partitionBy
но только для таблиц ( saveAsTable
). Информация о сегментировании может использоваться для оптимизации объединений:
// Temporarily disable broadcast joins
spark.conf.set("spark.sql.autoBroadcastJoinThreshold", -1)
df.write.bucketBy(42, "k").saveAsTable("df1")
val df2 = Seq(("A", -1.0), ("B", 2.0)).toDF("k", "v2")
df2.write.bucketBy(42, "k").saveAsTable("df2")
// == Physical Plan ==
// *Project [k#41, v#42, v2#47]
// +- *SortMergeJoin [k#41], [k#46], Inner
// :- *Sort [k#41 ASC NULLS FIRST], false, 0
// : +- *Project [k#41, v#42]
// : +- *Filter isnotnull(k#41)
// : +- *FileScan parquet default.df1[k#41,v#42] Batched: true, Format: Parquet, Location: InMemoryFileIndex[file:/spark-warehouse/df1], PartitionFilters: [], PushedFilters: [IsNotNull(k)], ReadSchema: struct<k:string,v:int>
// +- *Sort [k#46 ASC NULLS FIRST], false, 0
// +- *Project [k#46, v2#47]
// +- *Filter isnotnull(k#46)
// +- *FileScan parquet default.df2[k#46,v2#47] Batched: true, Format: Parquet, Location: InMemoryFileIndex[file:/spark-warehouse/df2], PartitionFilters: [], PushedFilters: [IsNotNull(k)], ReadSchema: struct<k:string,v2:double>
* Под разметкой разделов я подразумеваю только распределение данных. partitioned
RDD больше не имеет разделителя. ** Без предварительного прогноза. Если агрегирование охватывает только небольшое подмножество столбцов, вероятно, нет никакого выигрыша.