La soluzione accettata sarà estremamente lenta per molti dati. La soluzione con il maggior numero di voti è un po 'difficile da leggere e anche lenta con i dati numerici. Se ogni nuova colonna può essere calcolata indipendentemente dalle altre, le assegnerei direttamente senza utilizzarle apply.
Esempio con dati di personaggi falsi
Crea 100.000 stringhe in un DataFrame
df = pd.DataFrame(np.random.choice(['he jumped', 'she ran', 'they hiked'],
size=100000, replace=True),
columns=['words'])
df.head()
words
0 she ran
1 she ran
2 they hiked
3 they hiked
4 they hiked
Diciamo che volevamo estrarre alcune funzionalità di testo come fatto nella domanda originale. Ad esempio, estraiamo il primo carattere, contiamo la ricorrenza della lettera 'e' e capitalizziamo la frase.
df['first'] = df['words'].str[0]
df['count_e'] = df['words'].str.count('e')
df['cap'] = df['words'].str.capitalize()
df.head()
words first count_e cap
0 she ran s 1 She ran
1 she ran s 1 She ran
2 they hiked t 2 They hiked
3 they hiked t 2 They hiked
4 they hiked t 2 They hiked
Tempi
%%timeit
df['first'] = df['words'].str[0]
df['count_e'] = df['words'].str.count('e')
df['cap'] = df['words'].str.capitalize()
127 ms ± 585 µs per loop (mean ± std. dev. of 7 runs, 10 loops each)
def extract_text_features(x):
return x[0], x.count('e'), x.capitalize()
%timeit df['first'], df['count_e'], df['cap'] = zip(*df['words'].apply(extract_text_features))
101 ms ± 2.96 ms per loop (mean ± std. dev. of 7 runs, 10 loops each)
Sorprendentemente, puoi ottenere prestazioni migliori eseguendo il ciclo attraverso ogni valore
%%timeit
a,b,c = [], [], []
for s in df['words']:
a.append(s[0]), b.append(s.count('e')), c.append(s.capitalize())
df['first'] = a
df['count_e'] = b
df['cap'] = c
79.1 ms ± 294 µs per loop (mean ± std. dev. of 7 runs, 10 loops each)
Un altro esempio con dati numerici falsi
Crea 1 milione di numeri casuali e testa la powersfunzione dall'alto.
df = pd.DataFrame(np.random.rand(1000000), columns=['num'])
def powers(x):
return x, x**2, x**3, x**4, x**5, x**6
%%timeit
df['p1'], df['p2'], df['p3'], df['p4'], df['p5'], df['p6'] = \
zip(*df['num'].map(powers))
1.35 s ± 83.6 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
L'assegnazione di ogni colonna è 25 volte più veloce e molto leggibile:
%%timeit
df['p1'] = df['num'] ** 1
df['p2'] = df['num'] ** 2
df['p3'] = df['num'] ** 3
df['p4'] = df['num'] ** 4
df['p5'] = df['num'] ** 5
df['p6'] = df['num'] ** 6
51.6 ms ± 1.9 ms per loop (mean ± std. dev. of 7 runs, 10 loops each)
Ho fatto una risposta simile con maggiori dettagli qui sul perché in applygenere non è la strada da percorrere.
df.ix[: ,10:16]. Penso che dovrai inserire lemergetue funzionalità nel set di dati.