关注小编的公众号,后台回复“进群”,一起来交流学习吧!
前两篇中咱们分别介绍了使用Excel、Python和Hive SQL计算统计值,这次咱们使用Spark SQL来计算统计值。
先来回顾一下数据和对应的统计结果:
本文使用的是iris分类数据集,数据下载地址为:
http://archive.ics.uci.edu/ml/datasets/Iris
下载后转换为xlsx格式的文件,数据如下:
对应的统计结果如下:
在介绍之前,我还是想先说明一点,这一篇只是想先带大家体验一把Spark SQL,相关更多关于原理相关的知识,咱们会在后面的文章中详细介绍。
1、数据导入
这里咱们通过读取Excel的方式读取出相应的数据,并得到一个DataFrame:
def createDFByCSV(spark:SparkSession) = {
val df = spark.sqlContext.read.format("com.databricks.spark.csv")
.option("header","true") //这里如果在csv第一行有属性的话,没有就是"false"
.option("inferSchema",true.toString)//这是自动推断属性列的数据类型。
.load("resources/iris.csv")
df.show()
}
结果如下:
2、使用Spark SQL计算统计值
2.1 最大值、最小值
使用Spark SQL统计最大值或者最小值,首先使用agg函数对数据进行聚合,这个函数一般配合group by使用,不使用group by的话就相当于对所有的数据进行聚合。
随后,直接使用max和min函数就可以,想要输出多个结果的话,中间用逗号分开,而使用as给聚合后的结果赋予一个列名,相当于sql中的as:
import spark.implicits._
df.agg(max($"feature1") as "max_feature1",
min($"feature2") as "min_feature2")
.show()
结果输出如下:
上面的$代表一列的意思,相当于col函数:
df.agg(max(col("feature1")) as "max_feature1",
min(col("feature2")) as "min_feature2")
.show()
1.2 平均值
平均值的计算使用mean函数:
df.agg(mean($"feature1") as "mean_feature1",
mean($"feature2") as "mean_feature2").show()
输出为:
1.3 样本标准差&总体标准差
样本标准差的计算有两个函数可以使用,分别是stddev函数和stddev_samp函数,而总体标准差使用stddev_pop方法。需要注意的一点是,这里和hive sql是有区别的,在hive sql中,stddev函数代表的是总体标准差,而在spark sql中,stddev函数代表的是样本标准差,可以查看一下源代码:
通过代码验证一下:
df.agg(stddev($"feature1") as "stddev_feature1",
stddev_pop($"feature1") as "stddev_pop_feature1",
stddev_samp($"feature1") as "stddev_samp_feature1").show()
输出结果为:
1.4 中位数
SparkSQL中也没有直接计算中位数的方法,所以我们还是借鉴上一篇中的思路,再来回顾一下:
计算中位数也好,计算四分位数也好,无非就是要取得两个位置嘛,假设我们的数据从小到大排,按照1、2、3、.. 、n进行编号,当数量n为奇数时,取编号(n + 1)/2位置的数即可,当n为偶数时,取(int)(n + 1)/2位置和(int)(n + 1)/2 + 1位置的数取平均即可。但二者其实可以统一到一个公式中:
1)假设n = 149 ,(n+1)/2 = 75 ,小数部分为0,那么中位数=75位置的数 * (1 - 0)+ 76位置的数 * (0 - 0)
2)假设n = 150,(n+1)/2 = 75,小数部分为0.5,那么中位数=75位置的数 * (1 - 0.5)+ 76位置的数 * (0.5 - 0)
所以,可以把这个过程分解为三个步骤,第一步是给数字进行一个编号,spark中同样使用row_number()函数(该函数的具体用法后续再展开,这里只提供一个简单的例子),第二步是计算(n+1)/2的整数部分和小数部分,第三步就是根据公式计算中位数。
首先使用row_number()给数据进行编号:
val windowFun = Window.orderBy(col("feature3").asc)
df.withColumn("rank",row_number().over(windowFun)).show(false)
输出如下:
接下来是确定中位数的位置,这里我们分别拿到(n + 1)/2的整数部分和小数部分:
val median_index = df.agg(
((count($"feature3") + 1) / 2).cast("int") as "rank",
((count($"feature3") + 1) / 2 % 1) as "float_part"
)
median_index.show()
输出如下:
这里小数部分不为0,意味着我们不仅要拿到rank=75的数,还要拿到rank=76的数,我们最好把其放到一行上,这里使用同样lead函数,lead函数的作用就是拿到分组排序后,下一个位置或下n个位置的数,咱们在后面的博客中还会细讲,这里也只是抛砖引玉:
val windowFun = Window.orderBy(col("feature3").asc)
df.withColumn("next_feature3",lead(col("feature3"),1).over(windowFun)).show(false)
输出如下:
接下来,join两个表,按公式计算中位数就可以啦,完整的代码如下:
val median_index = df.agg(
((count($"feature3") + 1) / 2).cast("int") as "rank",
((count($"feature3") + 1) / 2 % 1) as "float_part"
)
val windowFun = Window.orderBy(col("feature3").asc)
df.withColumn("rank",row_number().over(windowFun))
.withColumn("next_feature3",lead(col("feature3"),1).over(windowFun))
.join(median_index,Seq("rank"),"inner")
.withColumn("median" ,($"float_part" - lit(0)) * $"next_feature3" + (lit(1) - $"float_part") * $"feature3")
.show()
输出如下:
1.5 四分位数
先来复习下四分位数的两种解法,n+1方法和n-1方法:
对于n+1方法,如果数据量为n,则四分位数的位置为:
Q1的位置= (n+1) × 0.25
Q2的位置= (n+1) × 0.5
Q3的位置= (n+1) × 0.75
对于n-1方法,如果数据量为n,则四分位数的位置为:
Q1的位置=1+(n-1)x 0.25
Q2的位置=1+(n-1)x 0.5
Q3的位置=1+(n-1)x 0.75
这里的思路和求解中位数是一样的,我们分别实现一下两种方法,首先是n+1方法:
val q1_index = df.agg(
((count($"feature3") + 1) * 0.25).cast("int") as "rank",
((count($"feature3") + 1) * 0.25 % 1) as "float_part"
)
val windowFun = Window.orderBy(col("feature3").asc)
df.withColumn("rank",row_number().over(windowFun))
.withColumn("next_feature3",lead(col("feature3"),1).over(windowFun))
.join(q1_index,Seq("rank"),"inner")
.withColumn("q1" ,($"float_part" - lit(0)) * $"next_feature3" + (lit(1) - $"float_part") * $"feature3")
.show()
输出为:
接下来是n-1方法:
val q1_index = df.agg(
((count($"feature3") - 1) * 0.25).cast("int") + 1 as "rank",
((count($"feature3") - 1) * 0.25 % 1) as "float_part"
)
val windowFun = Window.orderBy(col("feature3").asc)
df.withColumn("rank",row_number().over(windowFun))
.withColumn("next_feature3",lead(col("feature3"),1).over(windowFun))
.join(q1_index,Seq("rank"),"inner")
.withColumn("q1" ,($"float_part" - lit(0)) * $"next_feature3" + (lit(1) - $"float_part") * $"feature3")
.show()
输出为:
3、踩坑总结
在计算中位数或者四分位数时,我一开始的写法如下:
很奇怪的一点是,$"float_part" - 0没有报错,1 - $"float_part"却报错了,报的错误是:
看这里大家应该明白了,$"float_part" - 0中,减号左右两边的数据都应该是列名,与$"float_part" 类型相同,但是1 - $"float_part"两边都应该是个数字,与1的类型相同,所以后面一个报错了。
因此修改的方法是:
使用lit方法创建了一个全为0或者全为1的列,使得减号左右两边类型匹配。
网友评论