Tel*_*eba 5 python count pandas
我有一个数据框,其中记录了特定时间范围内几个人的速度。下面是一个简化版本:
df = pd.DataFrame([["Mary",0,2.3], ["Mary",1,1.8], ["Mary",2,3.2],
["Mary",3,3.0], ["Mary",4,2.6], ["Mary",5,2.2],
["Steve",0,1.6], ["Steve",1,1.7], ["Steve",2,2.5],
["Steve",3,2.7], ["Steve",4,2.3], ["Steve",5,1.8],
["Jane",0,1.9], ["Jane",1,2.7], ["Jane",2,2.3],
["Jane",3,1.9], ["Jane",4,2.2], ["Jane",5,2.1]],
columns = [ "name","time","speed (m/s)" ])
print(df)
name time (s) speed (m/s)
0 Mary 0 2.3
1 Mary 1 1.8
2 Mary 2 3.2
3 Mary 3 3.0
4 Mary 4 2.6
5 Mary 5 2.2
6 Steve 0 1.6
7 Steve 1 1.7
8 Steve 2 2.5
9 Steve 3 2.7
10 Steve 4 2.3
11 Steve 5 1.8
12 Jane 0 1.9
13 Jane 1 2.7
14 Jane 2 2.3
15 Jane 3 1.9
16 Jane 4 2.2
17 Jane 5 2.1
Run Code Online (Sandbox Code Playgroud)
我正在寻找一种方法来计算每个名称有多少次连续记录或更多记录的速度大于 2 m/s,以及这些流逝时间的平均持续时间。真实的数据帧有超过 150 万行,使得循环效率低下。
我期望的结果如下所示:
name count average_duration(s)
0 Mary 1 4 # from 2 to 5s (included) - 1 time, 4/1 = 4s
1 Steve 1 3 # from 2 to 4s (included) - 1 time, 3/1 = 3s
2 Jane 2 2 # from 1 to 2s & from 4 to 5s (included) - 2 times, 4/2 = 2s
Run Code Online (Sandbox Code Playgroud)
我在这个问题上花了一天多的时间,但没有成功......提前感谢您的帮助!
所以这是我的做法:
df['over2'] = df['speed (m/s)']>2
df['streak_id'] = (df['over2'] != df['over2'].shift(1)).cumsum()
streak_groups = df.groupby(['name','over2','streak_id'])["time"].agg(['min','max']).reset_index()
positive_streaks = streak_groups[streak_groups['over2'] & (streak_groups['min'] != streak_groups['max'])].copy()
positive_streaks["duration"] = positive_streaks["max"] - positive_streaks["min"] + 1
result = positive_streaks.groupby('name')['duration'].agg(['size', 'mean']).reset_index()
print(result)
Run Code Online (Sandbox Code Playgroud)
输出:
name size mean
0 Jane 2 2
1 Mary 1 4
2 Steve 1 3
Run Code Online (Sandbox Code Playgroud)
我基本上给每个 False/True 条纹一个唯一的 ID,以便能够按它进行分组,因此每个组都是一个连续的结果。
然后我简单地将持续时间作为最大时间-最小时间,去掉len 1的条纹,然后获取按名称分组的大小和平均值。
如果您想更好地理解每个步骤,我建议打印我一路上拥有的中间数据帧。