经典sql题(十三)炸裂对应学生的姓名和成绩

explodeposexplode 的区别

  1. explode:

    • 用于将数组中的每个元素展开为单独的行。
    • 结果中只包含元素的值,不包含其索引。
    • 如果输入数组有 n 个元素,结果将返回 n 行。
  2. posexplode:

    • 用于将数组中的每个元素展开为单独的行,同时提供每个元素的索引。
    • 结果包含两个列:一个是元素的索引,另一个是元素的值。
    • 如果输入数组有 n 个元素,结果将返回 n 行,但每行额外包含一个索引列。

示例对比

假设有一个数组 ['A', 'B', 'C']

使用 explode
SELECT explode(array['A', 'B', 'C']) AS element;

结果:

element
A
B
C

输出行数为 3(与数组元素数量相同)。

使用 posexplode
SELECT posexplode(array['A', 'B', 'C']) AS (index, element);

结果:

indexelement
0A
1B
2C

输出行数仍为 3,但除了元素值外,还附加了索引列。

总结

  • 行数: 两者在行数上是相同的(即每个数组元素生成一行)。
  • 输出内容: explode 只返回元素,而 posexplode 返回元素及其对应的索引。

1.explode 炸裂

如果需要同时拆分两个不对应的列(例如 studentscore),可以使用 explodesplit 结合 LATERAL VIEW,但这种方法不会保持行之间的对应关系,结果将会是笛卡尔积,即每个学生的名字都会与每个分数组合在一起。

示例数据

假设表格如下:

classstudentscore
1AAlice,Bob,Charlie90,85,80

使用 explode 同时拆分两个列

你可以使用以下 SQL 查询来拆分这两个列:

SELECT class, student_name, student_score
FROM test
LATERAL VIEW explode(split(student, ',')) AS student_name
LATERAL VIEW explode(split(score, ',')) AS student_score;

结果

由于直接使用 explode,结果将会是笛卡尔积,可能如下:

classstudent_namestudent_score
1AAlice90
1AAlice85
1AAlice80
1ABob90
1ABob85
1ABob80
1ACharlie90
1ACharlie85
1ACharlie80

缺点

  • 这种方法会导致所有学生与所有分数组合在一起,生成了一个完全的笛卡尔积。
  • 每个学生的名字都会与每个分数配对,而没有保持对应关系。

2.posexplode 炸裂

如果想要通过炸裂函数将两个列(如学生和分数)拆分并对应起来,可以使用 posexplode 对两个列进行操作,确保它们的索引相匹配。以下是如何实现这一点的详细说明。

使用 posexplode 对应两个列

你可以使用以下 SQL 查询将 studentscore 两列分别拆分,并确保它们的索引相对应。

SELECT class,
       student_name,
       student_score
FROM test
LATERAL VIEW posexplode(split(student, ',')) sn AS student_index, student_name
LATERAL VIEW posexplode(split(score, ',')) sc AS score_index, student_score
WHERE student_index = score_index;

解析

  1. split(student, ','):

    • student 列的字符串按 , 拆分为数组,例如 ['Alice', 'Bob', 'Charlie']
  2. posexplode(split(student, ',')):

    • 将拆分后的数组展开成多行,并为每个元素生成一个索引。例如:
      • student_index | student_name
      • 0 | Alice
      • 1 | Bob
      • 2 | Charlie
  3. split(score, ','):

    • 类似地将 score 列拆分为数组,例如 ['90', '85', '80']
  4. posexplode(split(score, ',')):

    • 将拆分后的分数数组也展开成多行,并生成索引。例如:
  • score_index | student_score
    • 0 | 90
    • 1 | 85
    • 2 | 80
  1. WHERE student_index = score_index:
    • 通过匹配两个索引,只保留对应的学生和分数。

结果

根据上述查询,输出将会是:

classstudent_namestudent_score
1AAlice90
1ABob85
1ACharlie80

通过这种方式,posexplode 可以有效地将两个列拆分并对应起来,确保每个学生的名字和分数在结果中正确匹配。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值