Как добавить отсутствующие даты в интервале дат? - PullRequest
1 голос
/ 24 сентября 2019

У меня есть фрейм данных, как показано ниже

df = pd.DataFrame({
'subject_id':[1,1,1,1,1,1,1,2,2,2,2,2],
'time_1' :['2173-04-03 12:35:00','2173-04-03 12:50:00','2173-04-05 
          12:59:00','2173-05-04 13:14:00','2173-05-05 13:37:00','2173-07-06 
          13:39:00','2173-07-08 11:30:00','2173-04-08 16:00:00','2173-04-09 
          22:00:00','2173-04-11 04:00:00','2173- 04-13 04:30:00','2173-04-14 08:00:00'],
 'val' :[5,5,5,5,1,6,5,5,8,3,4,6]})
 df['time_1'] = pd.to_datetime(df['time_1'])
 df['day'] = df['time_1'].dt.day
 df['month'] = df['time_1'].dt.month

Как видно из приведенного выше фрейма данных, между ними мало пропущенных дат. Я хотел бы создать новые записи для этих дат и заполнить значения непосредственно предыдущей строкой.

def dt(df):
   r = pd.date_range(start=df.date.min(), end=df.date.max())
   df.set_index('date').reindex(r)

new_df = df.groupby(['subject_id','month']).apply(dt)

. Создаются все даты.Я только хочу найти пропущенную дату в пределах интервала ввода даты для каждого предмета для каждого месяца

Я попробовал код из этой связанной записи .Хотя это помогло мне, но не дает ожидаемого результата для этого обновленного / нового требования.Поскольку мы оставляем соединение, оно копирует все записи.Я также не могу выполнить внутреннее объединение, потому что оно пропустит несоответствующий столбец.Я хочу сочетание левого и внутреннего соединения

В настоящее время он создает новые записи для всех 365 дней в году, которые я не хочу.что-то вроде ниже.Этого не ожидается

enter image description here

Я только хочу добавить пропущенные даты между входным интервалом дат, как показано ниже.Например subject = 1, в 4-м месяце есть записи с 3-го и 5-го.но 4-й отсутствует.Итак, мы добавляем запись только за 4-й день.Нам не нужны 6-е, 7-е и т. Д., В отличие от токового выхода.Аналогично в 7-м месяце пропадает запись за 7-й день.поэтому мы просто добавляем новую запись для этого

Я ожидаю, что мой вывод будет таким, как показано ниже

enter image description here

Ответы [ 2 ]

3 голосов
/ 24 сентября 2019

Вот проблема, которая вам нужна resample для добавления новых дней, поэтому это необходимо.

df['time_1'] = pd.to_datetime(df['time_1'])
df['day'] = df['time_1'].dt.day
df['date'] = df['time_1'].dt.floor('d')

df1 = (df.set_index('date')
         .groupby('subject_id')
         .resample('d')
         .last()
         .index
         .to_frame(index=False))
print (df1)
     subject_id       date
0             1 2173-04-03
1             1 2173-04-04
2             1 2173-04-05
3             1 2173-04-06
4             1 2173-04-07
..          ...        ...
99            2 2173-04-10
100           2 2173-04-11
101           2 2173-04-12
102           2 2173-04-13
103           2 2173-04-14

[104 rows x 2 columns]

Идея состоит в том, чтобы удалить ненужные пропущенные строки - вы можете создать порог для минимальных последовательных значений затуманивания (здесь 5)и удалить строки (создан новый столбец для простого теста):

df2 = df1.merge(df, how='left') 

thresh = 5
mask = df2['day'].notna()
s = mask.cumsum().mask(mask)
df2['count'] = s.map(s.value_counts())

df2 = df2[(df2['count'] < thresh) | (df2['count'].isna())]
print (df2)
     subject_id       date              time_1  val   day  count
0             1 2173-04-03 2173-04-03 12:35:00  5.0   3.0    NaN
1             1 2173-04-03 2173-04-03 12:50:00  5.0   3.0    NaN
2             1 2173-04-04                 NaT  NaN   NaN    1.0
3             1 2173-04-05 2173-04-05 12:59:00  5.0   5.0    NaN
32            1 2173-05-04 2173-05-04 13:14:00  5.0   4.0    NaN
33            1 2173-05-05 2173-05-05 13:37:00  1.0   5.0    NaN
95            1 2173-07-06 2173-07-06 13:39:00  6.0   6.0    NaN
96            1 2173-07-07                 NaT  NaN   NaN    1.0
97            1 2173-07-08 2173-07-08 11:30:00  5.0   8.0    NaN
98            2 2173-04-08 2173-04-08 16:00:00  5.0   8.0    NaN
99            2 2173-04-09 2173-04-09 22:00:00  8.0   9.0    NaN
100           2 2173-04-10                 NaT  NaN   NaN    1.0
101           2 2173-04-11 2173-04-11 04:00:00  3.0  11.0    NaN
102           2 2173-04-12                 NaT  NaN   NaN    1.0
103           2 2173-04-13 2173-04-13 04:30:00  4.0  13.0    NaN
104           2 2173-04-14 2173-04-14 08:00:00  6.0  14.0    NaN

Последнее использование предыдущего решения:

df2 = df2.groupby(df['subject_id']).ffill()
dates = df2['time_1'].dt.normalize() 
df2['time_1'] += np.where(dates == df2['date'], 0, df2['date'] - dates)
df2['day'] = df2['time_1'].dt.day
df2['val'] = df2['val'].astype(int)
print (df2)
     subject_id       date              time_1  val  day  count
0             1 2173-04-03 2173-04-03 12:35:00    5    3    NaN
1             1 2173-04-03 2173-04-03 12:50:00    5    3    NaN
2             1 2173-04-04 2173-04-04 12:50:00    5    4    1.0
3             1 2173-04-05 2173-04-05 12:59:00    5    5    1.0
32            1 2173-05-04 2173-05-04 13:14:00    5    4    NaN
33            1 2173-05-05 2173-05-05 13:37:00    1    5    NaN
95            1 2173-07-06 2173-07-06 13:39:00    6    6    NaN
96            1 2173-07-07 2173-07-07 13:39:00    6    7    1.0
97            1 2173-07-08 2173-07-08 11:30:00    5    8    1.0
98            2 2173-04-08 2173-04-08 16:00:00    5    8    1.0
99            2 2173-04-09 2173-04-09 22:00:00    8    9    1.0
100           2 2173-04-10 2173-04-10 22:00:00    8   10    1.0
101           2 2173-04-11 2173-04-11 04:00:00    3   11    1.0
102           2 2173-04-12 2173-04-12 04:00:00    3   12    1.0
103           2 2173-04-13 2173-04-13 04:30:00    4   13    1.0
104           2 2173-04-14 2173-04-14 08:00:00    6   14    1.0

РЕДАКТИРОВАТЬ: Решение с reindex для каждого месяца:

df['time_1'] = pd.to_datetime(df['time_1'])
df['day'] = df['time_1'].dt.day
df['date'] = df['time_1'].dt.floor('d')
df['month'] = df['time_1'].dt.month

df1 = (df.drop_duplicates(['date','subject_id'])
          .set_index('date')
         .groupby(['subject_id', 'month'])
         .apply(lambda x: x.reindex(pd.date_range(x.index.min(), x.index.max())))
         .rename_axis(('subject_id','month','date'))
         .index
         .to_frame(index=False)
         )
print (df1)
    subject_id  month       date
0            1      4 2173-04-03
1            1      4 2173-04-04
2            1      4 2173-04-05
3            1      5 2173-05-04
4            1      5 2173-05-05
5            1      7 2173-07-06
6            1      7 2173-07-07
7            1      7 2173-07-08
8            2      4 2173-04-08
9            2      4 2173-04-09
10           2      4 2173-04-10
11           2      4 2173-04-11
12           2      4 2173-04-12
13           2      4 2173-04-13
14           2      4 2173-04-14

df2 = df1.merge(df, how='left') 
df2 = df2.groupby(df2['subject_id']).ffill()
dates = df2['time_1'].dt.normalize() 
df2['time_1'] += np.where(dates == df2['date'], 0, df2['date'] - dates)
df2['day'] = df2['time_1'].dt.day
df2['val'] = df2['val'].astype(int)
print (df2)
    subject_id  month       date              time_1  val  day
0            1      4 2173-04-03 2173-04-03 12:35:00    5    3
1            1      4 2173-04-03 2173-04-03 12:50:00    5    3
2            1      4 2173-04-04 2173-04-04 12:50:00    5    4
3            1      4 2173-04-05 2173-04-05 12:59:00    5    5
4            1      5 2173-05-04 2173-05-04 13:14:00    5    4
5            1      5 2173-05-05 2173-05-05 13:37:00    1    5
6            1      7 2173-07-06 2173-07-06 13:39:00    6    6
7            1      7 2173-07-07 2173-07-07 13:39:00    6    7
8            1      7 2173-07-08 2173-07-08 11:30:00    5    8
9            2      4 2173-04-08 2173-04-08 16:00:00    5    8
10           2      4 2173-04-09 2173-04-09 22:00:00    8    9
11           2      4 2173-04-10 2173-04-10 22:00:00    8   10
12           2      4 2173-04-11 2173-04-11 04:00:00    3   11
13           2      4 2173-04-12 2173-04-12 04:00:00    3   12
14           2      4 2173-04-13 2173-04-13 04:30:00    4   13
15           2      4 2173-04-14 2173-04-14 08:00:00    6   14
1 голос
/ 24 сентября 2019

Помогает ли это?

def fill_dates(df):
    result = pd.DataFrame()
    for i,row in df.iterrows():
        if i == 0:
            result = result.append(row)
        else:
            start_date = result.iloc[-1]['time_1']
            end_date = row['time_1']
#            print(start_date, end_date)
            delta = (end_date - start_date).days
#            print(delta)
            if delta > 0 and start_date.month == end_date.month:
                for j in range(delta):
                    day = start_date + timedelta(days=j+1)
                    new_row = result.iloc[-1].copy()
                    new_row['time_1'] = day
                    new_row['remarks'] = 'added'
                    if new_row['time_1'].date() != row['time_1'].date():
                        result = result.append(new_row)
                result = result.append(row)
            else:
                result = result.append(row)
    result.reset_index(inplace = True)
    return result
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...