gpt4 book ai didi

json - 如何使用 pyspark explode() 分解结构

转载 作者:行者123 更新时间:2023-12-05 03:45:05 24 4
gpt4 key购买 nike

如何将以下 JSON 转换为它后面的关系行?我坚持的部分是 pyspark explode() 函数由于类型不匹配而抛出异常。我还没有找到一种方法来将数据强制转换为合适的格式,以便我可以从 sample_json 对象中的 source 键中的每个对象创建行。

JSON 输入

sample_json = """
{
"dc_id": "dc-101",
"source": {
"sensor-igauge": {
"id": 10,
"ip": "68.28.91.22",
"description": "Sensor attached to the container ceilings",
"temp":35,
"c02_level": 1475,
"geo": {"lat":38.00, "long":97.00}
},
"sensor-ipad": {
"id": 13,
"ip": "67.185.72.1",
"description": "Sensor ipad attached to carbon cylinders",
"temp": 34,
"c02_level": 1370,
"geo": {"lat":47.41, "long":-122.00}
},
"sensor-inest": {
"id": 8,
"ip": "208.109.163.218",
"description": "Sensor attached to the factory ceilings",
"temp": 40,
"c02_level": 1346,
"geo": {"lat":33.61, "long":-111.89}
},
"sensor-istick": {
"id": 5,
"ip": "204.116.105.67",
"description": "Sensor embedded in exhaust pipes in the ceilings",
"temp": 40,
"c02_level": 1574,
"geo": {"lat":35.93, "long":-85.46}
}
}
}"""

期望的输出

dc_id    source_name    id    description
-------------------------------------------------------------------------------
dc-101 sensor-gauge 10 Sensor attached to the container ceilings
dc-101 sensor-ipad 13 Sensor ipad attached to carbon cylinders
dc-101 sensor-inest 8 Sensor attached to the factory ceilings
dc-101 sensor-istick 5 Sensor embedded in exhaust pipes in the ceilings

PYSPARK 代码

from pyspark.sql.functions import *
df_sample_data = spark.read.json(sc.parallelize([sample_json]))
df_expanded = df_sample_data.withColumn("one_source",explode_outer(col("source")))
display(df_expanded)

错误

AnalysisException: cannot resolve 'explode(source)' due to data typemismatch: input to function explode should be array or map type, notstruct....

我把这个放在一起 Databricks notebook进一步证明挑战并清楚地显示错误。我将能够使用此笔记本来测试此处提供的任何建议。

最佳答案

您不能对结构使用 explode 但您可以在结构 source 中获取列名(使用 df.select("source.*").columns) 并使用列表理解从每个嵌套结构中创建所需字段的数组,然后展开以获得所需的结果:

from pyspark.sql import functions as F

df1 = df.select(
"dc_id",
F.explode(
F.array(*[
F.struct(
F.lit(s).alias("source_name"),
F.col(f"source.{s}.id").alias("id"),
F.col(f"source.{s}.description").alias("description")
)
for s in df.select("source.*").columns
])
).alias("sources")

).select("dc_id", "sources.*")

df1.show(truncate=False)

#+------+-------------+---+------------------------------------------------+
#|dc_id |source_name |id |description |
#+------+-------------+---+------------------------------------------------+
#|dc-101|sensor-igauge|10 |Sensor attached to the container ceilings |
#|dc-101|sensor-inest |8 |Sensor attached to the factory ceilings |
#|dc-101|sensor-ipad |13 |Sensor ipad attached to carbon cylinders |
#|dc-101|sensor-istick|5 |Sensor embedded in exhaust pipes in the ceilings|
#+------+-------------+---+------------------------------------------------+

关于json - 如何使用 pyspark explode() 分解结构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66130815/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com