Ste*_* Li 49 scala apache-spark apache-spark-sql spark-dataframe
我使用以下模式在spark中创建了一个数据框:
root
|-- user_id: long (nullable = false)
|-- event_id: long (nullable = false)
|-- invited: integer (nullable = false)
|-- day_diff: long (nullable = true)
|-- interested: integer (nullable = false)
|-- event_owner: long (nullable = false)
|-- friend_id: long (nullable = false)
Run Code Online (Sandbox Code Playgroud)
数据如下所示:
+----------+----------+-------+--------+----------+-----------+---------+
| user_id| event_id|invited|day_diff|interested|event_owner|friend_id|
+----------+----------+-------+--------+----------+-----------+---------+
| 4236494| 110357109| 0| -1| 0| 937597069| null|
| 78065188| 498404626| 0| 0| 0| 2904922087| null|
| 282487230|2520855981| 0| 28| 0| 3749735525| null|
| 335269852|1641491432| 0| 2| 0| 1490350911| null|
| 437050836|1238456614| 0| 2| 0| 991277599| null|
| 447244169|2095085551| 0| -1| 0| 1579858878| null|
| 516353916|1076364848| 0| 3| 1| 3597645735| null|
| 528218683|1151525474| 0| 1| 0| 3433080956| null|
| 531967718|3632072502| 0| 1| 0| 3863085861| null|
| 627948360|2823119321| 0| 0| 0| 4092665803| null|
| 811791433|3513954032| 0| 2| 0| 415464198| null|
| 830686203| 99027353| 0| 0| 0| 3549822604| null|
|1008893291|1115453150| 0| 2| 0| 2245155244| null|
|1239364869|2824096896| 0| 2| 1| 2579294650| null|
|1287950172|1076364848| 0| 0| 0| 3597645735| null|
|1345896548|2658555390| 0| 1| 0| 2025118823| null|
|1354205322|2564682277| 0| 3| 0| 2563033185| null|
|1408344828|1255629030| 0| -1| 1| 804901063| null|
|1452633375|1334001859| 0| 4| 0| 1488588320| null|
|1625052108|3297535757| 0| 3| 0| 1972598895| null|
+----------+----------+-------+--------+----------+-----------+---------+
Run Code Online (Sandbox Code Playgroud)
我想过滤掉"friend_id"字段中的行具有空值.
scala> val aaa = test.filter("friend_id is null")
scala> aaa.count
Run Code Online (Sandbox Code Playgroud)
我得到了:res52:Long = 0这显然不对.获得它的正确方法是什么?
还有一个问题,我想替换friend_id字段中的值.我想用0和1替换null,除了null之外的任何其他值.我能弄清楚的代码是:
val aaa = train_friend_join.select($"user_id", $"event_id", $"invited", $"day_diff", $"interested", $"event_owner", ($"friend_id" != null)?1:0)
Run Code Online (Sandbox Code Playgroud)
此代码也不起作用.任何人都可以告诉我如何解决它?谢谢
Sac*_*agi 61
假设你有这个数据设置(这样结果可以重现):
// declaring data types
case class Company(cName: String, cId: String, details: String)
case class Employee(name: String, id: String, email: String, company: Company)
// setting up example data
val e1 = Employee("n1", null, "n1@c1.com", Company("c1", "1", "d1"))
val e2 = Employee("n2", "2", "n2@c1.com", Company("c1", "1", "d1"))
val e3 = Employee("n3", "3", "n3@c1.com", Company("c1", "1", "d1"))
val e4 = Employee("n4", "4", "n4@c2.com", Company("c2", "2", "d2"))
val e5 = Employee("n5", null, "n5@c2.com", Company("c2", "2", "d2"))
val e6 = Employee("n6", "6", "n6@c2.com", Company("c2", "2", "d2"))
val e7 = Employee("n7", "7", "n7@c3.com", Company("c3", "3", "d3"))
val e8 = Employee("n8", "8", "n8@c3.com", Company("c3", "3", "d3"))
val employees = Seq(e1, e2, e3, e4, e5, e6, e7, e8)
val df = sc.parallelize(employees).toDF
Run Code Online (Sandbox Code Playgroud)
数据是:
+----+----+---------+---------+
|name| id| email| company|
+----+----+---------+---------+
| n1|null|n1@c1.com|[c1,1,d1]|
| n2| 2|n2@c1.com|[c1,1,d1]|
| n3| 3|n3@c1.com|[c1,1,d1]|
| n4| 4|n4@c2.com|[c2,2,d2]|
| n5|null|n5@c2.com|[c2,2,d2]|
| n6| 6|n6@c2.com|[c2,2,d2]|
| n7| 7|n7@c3.com|[c3,3,d3]|
| n8| 8|n8@c3.com|[c3,3,d3]|
+----+----+---------+---------+
Run Code Online (Sandbox Code Playgroud)
现在用过滤器过滤员工null,你会做 -
df.filter("id is null").show
Run Code Online (Sandbox Code Playgroud)
这将正确显示以下内容:
+----+----+---------+---------+
|name| id| email| company|
+----+----+---------+---------+
| n1|null|n1@c1.com|[c1,1,d1]|
| n5|null|n5@c2.com|[c2,2,d2]|
+----+----+---------+---------+
Run Code Online (Sandbox Code Playgroud)
来到你的问题的第二部分,你可以null用0 替换id为0和其他值为1 -
df.withColumn("id", when($"id".isNull, 0).otherwise(1)).show
Run Code Online (Sandbox Code Playgroud)
这导致:
+----+---+---------+---------+
|name| id| email| company|
+----+---+---------+---------+
| n1| 0|n1@c1.com|[c1,1,d1]|
| n2| 1|n2@c1.com|[c1,1,d1]|
| n3| 1|n3@c1.com|[c1,1,d1]|
| n4| 1|n4@c2.com|[c2,2,d2]|
| n5| 0|n5@c2.com|[c2,2,d2]|
| n6| 1|n6@c2.com|[c2,2,d2]|
| n7| 1|n7@c3.com|[c3,3,d3]|
| n8| 1|n8@c3.com|[c3,3,d3]|
+----+---+---------+---------+
Run Code Online (Sandbox Code Playgroud)
chA*_*xey 16
我找到了一个很好的解决方案,可以删除任何空值的行:
Dataset<Row> filtered = df.filter(row => !row.anyNull);
如果一个人对另一个案件感兴趣,请打电话row.anyNull.(使用Java API的Spark 2.1.0)
Ayu*_*yan 13
有两种方法可以做到这一点:创建过滤条件1)手动2)动态.
示例DataFrame:
val df = spark.createDataFrame(Seq(
(0, "a1", "b1", "c1", "d1"),
(1, "a2", "b2", "c2", "d2"),
(2, "a3", "b3", null, "d3"),
(3, "a4", null, "c4", "d4"),
(4, null, "b5", "c5", "d5")
)).toDF("id", "col1", "col2", "col3", "col4")
+---+----+----+----+----+
| id|col1|col2|col3|col4|
+---+----+----+----+----+
| 0| a1| b1| c1| d1|
| 1| a2| b2| c2| d2|
| 2| a3| b3|null| d3|
| 3| a4|null| c4| d4|
| 4|null| b5| c5| d5|
+---+----+----+----+----+
Run Code Online (Sandbox Code Playgroud)
1)手动创建过滤条件,即使用DataFrame where或filter函数
df.filter(col("col1").isNotNull && col("col2").isNotNull).show
Run Code Online (Sandbox Code Playgroud)
要么
df.where("col1 is not null and col2 is not null").show
Run Code Online (Sandbox Code Playgroud)
结果:
+---+----+----+----+----+
| id|col1|col2|col3|col4|
+---+----+----+----+----+
| 0| a1| b1| c1| d1|
| 1| a2| b2| c2| d2|
| 2| a3| b3|null| d3|
+---+----+----+----+----+
Run Code Online (Sandbox Code Playgroud)
2)动态创建过滤条件:当我们不希望任何列具有空值并且列数很多(大多数情况下是这种情况)时,这很有用.
在这些情况下手动创建过滤条件会浪费大量时间.在下面的代码中,我们包括动态使用map和reduce在DataFrame列上运行的所有列:
val filterCond = df.columns.map(x=>col(x).isNotNull).reduce(_ && _)
Run Code Online (Sandbox Code Playgroud)
filterCond看起来如何:
filterCond: org.apache.spark.sql.Column = (((((id IS NOT NULL) AND (col1 IS NOT NULL)) AND (col2 IS NOT NULL)) AND (col3 IS NOT NULL)) AND (col4 IS NOT NULL))
Run Code Online (Sandbox Code Playgroud)
过滤:
val filteredDf = df.filter(filterCond)
Run Code Online (Sandbox Code Playgroud)
结果:
+---+----+----+----+----+
| id|col1|col2|col3|col4|
+---+----+----+----+----+
| 0| a1| b1| c1| d1|
| 1| a2| b2| c2| d2|
+---+----+----+----+----+
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
118115 次 |
| 最近记录: |