Beschreibende Datenanalyse#

KISS - Keep it Simple Stupid

— Kelly Johnson

Der erste Schritt beim Analysieren eines neuen Datensatzes ist es sich mit den Daten vertraut zu machen. Hierbei ist es noch nicht Ziel aus den Daten Statistiken zu extrahieren, sie zu Visualisieren oder zu Säubern, was wir in separaten Abschnitten behandeln. Stattdessen versucht man sich einen Überblick zu verschaffen: Welche Spalten gibt es? Gibt es fehlende Werte? Wie sind die Einträge verteilt?

Gerade im Bauwesen, z. B. bei Sensor- oder Baudaten, ist das entscheidend, um zu erkennen, ob die Daten überhaupt für eine Modellierung geeignet sind.

Folien#

Erste Schritte: Kenne deine Daten#

Als nächstes machen wir einige Beispiele für explorative Datenanalyse mit Pandas. Hierfür laden wir als Beispiel die Liste aller aktuellen Baustellen in Rostock an, die von der Stadt Rostock im Open Data Portal zur Verfügung gestellt wird. Öffentliche Daten wie diese zeigen reale Bauaktivitäten und die Analyse hilft z.B. bei der Planung, Verkehrsführung oder Risikobewertung bei parallelen Projekten. Solche Daten können später genutzt werden, um Modelle zur Vorhersage von Bauverzögerungen, Verkehrsbelastung oder Personalplanung zu trainieren.

Wir lesen die komprimierte baustdatendatei direkt in Pandas ein. Da diese durch Komma getrennt ist, müssen keinen Trennzeichen angeben, da das Komma der Standard-Trennzeichen ist.

import numpy as np # Import von NumPy
import pandas as pd # Import von Pandas

baust  = pd.read_csv("../data/Baustellen/baustellen_flat.csv")

Als ersten Schritt können wir die Größe des Datensatzes abfragen. Datenrahmen haben ein Attribut .shape, genau wie numpy-Arrays:

baust.shape
(204, 19)

Die Antwort sagt uns die Anzahl der Zeilen und Spalten an. Z. B. (204, 19) bedeutet 204 Zeilen mit je 19 Spalten. Das hilft abzuschätzen, wie umfangreich der Datensatz ist und ob er vollständig erscheint.

Als nächstes könnten wir einen schnellen Blick auf ein paar Zeilen der Daten werfen. Die Attribute .head() und .tail() sind nützlich, um sich die ersten und letzten Zeilen anzusehen, .sample() extrahiert ein paar zufällige Zeilen:

baust.head()
latitude longitude uuid kreis_name kreis_schluessel gemeindeverband_name gemeindeverband_schluessel gemeinde_name gemeinde_schluessel strasse_name strasse_schluessel sparte von nach baubeginn bauende verkehrsbeeintraechtigungen baumassnahme dauer
0 54.089282 12.111994 063a229c-db25-45af-9fba-8d963b287910 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Budapester Str. 01640 Fernwärmeleitung 21 NaN 2024-03-04 07:00:00+01:00 2024-04-12 17:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Verkeh... FW-Leitung i.a. der Hanseatic 39.375000
1 54.068210 12.078264 07165e95-4e5f-429c-88b6-125d261cbcda Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Satower Str. 08180 Grünpflege 56 65 2024-03-07 07:00:00+01:00 2024-03-31 15:00:00+02:00 halbseitige Sperrung, Sicherungsmaßnahmen entl... Baumpflanzung 24.291667
2 54.174303 12.081928 077f9448-c35e-4074-8491-7d142045a8db Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Am Markt 00460 Wasserleitung 6 NaN 2024-03-06 07:00:00+01:00 2024-03-28 18:00:00+01:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Einbau Trinkwasserschieber 22.458333
3 54.082628 12.126542 0c04e4b5-6e82-49b6-8b60-1c2cf9c1b1cf Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Klopstockstr. 06070 Gebäudesanierung 5 NaN 2024-01-08 00:00:00+01:00 2024-03-30 00:00:00+01:00 NaN NaN 82.000000
4 54.074173 12.123860 0c122d6b-a6bd-4c41-b25b-09fe10a23a2e Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Ziolkowskistr. 09920 Wasserleitung NaN NaN 2023-03-06 00:00:00+01:00 2024-08-01 00:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Sanierung Mischwasser- und Trinkwasserleitung 513.958333
baust.tail(3)
latitude longitude uuid kreis_name kreis_schluessel gemeindeverband_name gemeindeverband_schluessel gemeinde_name gemeinde_schluessel strasse_name strasse_schluessel sparte von nach baubeginn bauende verkehrsbeeintraechtigungen baumassnahme dauer
201 54.080468 12.126524 fd2e99b4-9ac8-4dd2-94c2-b521b6d91455 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Südring 07400 Gleisbau Platz der Freundschaft (07650) Goetheplatz (02960) 2022-05-04 00:00:00+02:00 2025-01-01 00:00:00+01:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Baustelleneinrichtung / Belieferung für Neubau... 973.041667
202 54.151488 12.080890 fd32f630-333c-4752-9989-8c41a8eb5b30 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Taklerring 08880 Hochbau 45 NaN 2023-06-01 00:00:00+02:00 2025-08-31 00:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Baustraße für Neubau Hort Taklerring 822.000000
203 54.086911 12.104815 fdc7ab16-67ac-4d33-ae98-b0bbd9703f18 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Dethardingstr. 04330 Telefonnetz 18 29 2024-03-25 07:00:00+01:00 2024-03-28 18:00:00+01:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Breitbandausbau 3.458333
baust.sample(4)
latitude longitude uuid kreis_name kreis_schluessel gemeindeverband_name gemeindeverband_schluessel gemeinde_name gemeinde_schluessel strasse_name strasse_schluessel sparte von nach baubeginn bauende verkehrsbeeintraechtigungen baumassnahme dauer
186 54.123774 12.054765 ecabda72-7bb0-4ffd-82e1-1de9c6c83db0 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Willi-Bredel-Str. 09820 Straßenbeleuchtung 13 23 2024-02-26 07:00:00+01:00 2024-04-12 18:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Erneuerung Straßenbeleuchtung 46.416667
100 54.083103 12.128079 92342d86-767e-490c-9c7b-3c38114efb18 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 St.-Georg-Str. 02590 Hochbau 50 NaN 2024-02-19 00:00:00+01:00 2025-01-01 00:00:00+01:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Lieferverkehr und Materiallagerung für Hochbau 317.000000
158 54.087803 12.117272 d2d9d283-90de-4936-94f0-a6002d09533d Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Barnstorfer Weg 01120 Gebäudesanierung 21 NaN 2024-02-01 00:00:00+01:00 2024-04-13 00:00:00+02:00 NaN NaN 71.958333
179 54.089161 12.118224 e8752e43-3b4f-461b-a26b-26dcc487d2ea Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Leonhardstr. 06550 Straßenbau 10 NaN 2024-03-14 07:00:00+01:00 2024-04-12 18:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Reparatur Gehweg 29.416667

Diese Funktionen zeigt uns zwar im ersten Eindruck viele Spalten, aber nicht immer alle, wenn der Bildschirmplatz nicht ausreicht. Deshalb ist es bei solchen Datensätzen mit vielen Spalten sinnvoll alle Spaltennamen in einem Datensatz aufzulisten.

baust.columns
Index(['latitude', 'longitude', 'uuid', 'kreis_name', 'kreis_schluessel',
       'gemeindeverband_name', 'gemeindeverband_schluessel', 'gemeinde_name',
       'gemeinde_schluessel', 'strasse_name', 'strasse_schluessel', 'sparte',
       'von', 'nach', 'baubeginn', 'bauende', 'verkehrsbeeintraechtigungen',
       'baumassnahme', 'dauer'],
      dtype='object')

Als nächstes überprüfen wir die Datentypen der Daten. Obwohl einige, z.B. kreis_name und baumassnahme aussehen wie ein String, könnten sie tatsächlich einen anderen Typ haben. Dies kann über das Attribut .dtypes abgefragt werden:

baust.dtypes
latitude                       float64
longitude                      float64
uuid                            object
kreis_name                      object
kreis_schluessel               float64
gemeindeverband_name            object
gemeindeverband_schluessel     float64
gemeinde_name                   object
gemeinde_schluessel            float64
strasse_name                    object
strasse_schluessel              object
sparte                          object
von                             object
nach                            object
baubeginn                       object
bauende                         object
verkehrsbeeintraechtigungen     object
baumassnahme                    object
dauer                          float64
dtype: object

Die Datentypen geben an, dass sowohl kreis_name und baumassnahme keine Strings sondern Objekte sind. Hier bedeutet es Zeichenfolgen, im Prinzip können es auch komplexere Objekte sein. Wenn wir zum Beispiel mal die Anzahl der Werte für kreis_name berechnen

baust.kreis_name.count()
203

so ist diese Zahl niedriger als die Anzahl der Zeilen die baust.shape zurückgeliefert hat. Das liegt an fehlenden Werten. Dazu mehr im Abschnitt Datenvorverarbeitung.

Was sind die Werte?#

Einer der ersten Schritte mit einem neuen Datensatz ist es, die Werte der relevanten Variablen zu überprüfen. Die Pandas-Bibliothek enthält viele Tools für die beschreibenden Datenanalyse. Bei den kategorischen Variablen möchten wir normalerweise die expliziten Werte sehen, bei den numerischen können wir die minimalen und maximalen Werte überprüfen.

Verschiedene diskrete Werte#

Was sind die möglichen Werte für Kreisname? Ein schneller Blick auf die Daten zeigt, dass es einige Werte für die Spalte sparte, aber gibt es noch mehr? Das können wir mit der Methode .unique() abfragen:

baust.sparte.unique()
array(['Fernwärmeleitung', 'Grünpflege', 'Wasserleitung',
       'Gebäudesanierung', 'Hochbau', 'Straßenbau', 'Stromnetz',
       'Kabelnetz', 'Gasleitung', 'Telefonnetz', 'Lichtsignalanlage',
       'Straßenbeleuchtung', 'privat', 'Gleisbau', 'Kranarbeiten'],
      dtype=object)

Das Ergebnis zeigt uns, dass alle Passagiere in diesen Daten als “männlich” oder “weiblich” kategorisiert sind. Das scheint vollkommen plausibel zu sein. Wir können auch ein technisches Detail erkennen, nämlich dass das Ergebnis als ein numpy-Array zurückgegeben wird.

Tip

Ein Wort über Methoden und Methodenverkettung. Lassen Sie uns den vorherigen Befehl, baust.sparte.unique(), überprüfen. Dieser enthält drei Komponenten, und der Prozess besteht darin, diese in einer sequenziellen Reihenfolge anzuwenden:

  • baust ist der Datensatz. Das ist das, womit wir anfangen.

  • .sparte ist ein Attribut von baust. Hier bedeutet es, das die Spalte sparte aus dem Datensatz extrahiert wird und als Serie zurückgegeben wird.

  • Schließlich ist .unique() eine Methode, welche die Serie an Werten auf ihrer linken Seite analysiert (hier die Serie sparte), und diese analysiert um die eindeutigen Werten zu identifizieren.

Einige der Methoden können entweder auf den gesamten Datenrahmen oder auf einzelne Variablen angewendet werden. Wenn sie auf den gesamten Datenrahmen angewendet werden, gelten sie für jede einzelne Variable separat, sozusagen wie eine Schleife über einzelne Variablen, auf die sie angewendet werden.

Häufig ist es nicht nur interessant bei kategorischen Werten zu verstehen, welche eindeutigen Werte auftreten, sondern wie oft diese Auftreten. Hierfür nutzt man die Funktion

baust.sparte.value_counts()
sparte
Hochbau               40
Gebäudesanierung      35
Straßenbau            35
Wasserleitung         25
Fernwärmeleitung      18
Telefonnetz           16
Kabelnetz             11
Stromnetz              8
Grünpflege             4
Lichtsignalanlage      3
Straßenbeleuchtung     3
Gleisbau               2
Kranarbeiten           2
Gasleitung             1
privat                 1
Name: count, dtype: int64

Das gibt uns schon mal ein gutes Bild der Sparten. Aber wie sieht es mit der Spalte baumassnahme aus?

baust.baumassnahme.unique()
array(['FW-Leitung i.a. der Hanseatic', 'Baumpflanzung',
       'Einbau Trinkwasserschieber', nan,
       'Sanierung Mischwasser- und Trinkwasserleitung',
       'Containerstellung', 'Kennzeichnung Baustellenausfahrten',
       'Wartungs- und Instandsetzungsarbeiten innerhalb der konzessionierten Strecke',
       'Grünraumarbeiten, Sanierung Außenanlagen',
       'Ausschleifung - Muffung, StS Bussebart', 'Sanierungsarbeiten',
       'Schutzrohrverlegung', 'Herstellung Gehweg und Parkflächen',
       'Verlegung Fernwärme', 'Fernwärmeerweiterung',
       'LWL- Verlegung im Gehweg und Grünstreifen',
       'Reparatur Trinkwasserschieber',
       'Ausbau Gewerbestraße mit Zufahrt CNG-Tankstelle', 'Straßenneubau',
       'Tiefbauarbeiten für Breitbandausbau', 'Breitbandausbau',
       'Sanierung Gehweg', 'Neubau Trinkwassergrundstücksanschluss',
       'Fernwärme', 'Sanierung Altbau', 'Herstellung Aufpflasterung',
       'Wartungs- und Instandsetzungsarbeiten an Wechselverkehrszeichen (WVZ) - AQ0 und AQ1',
       'Neubau Wohnhaus', 'Hausanschluss Fernwärme',
       'Gehwegsanierung und Verkehrsberuhigung',
       'Neubau Fernwärmeleitung -abschnittsweise-', 'Gehwegsanierung',
       'Hochbauarbeiten', 'Reparatur Hauptleitung Fernwärme',
       'Verlegung LWL', 'Neubau Mehrfamilienhaus',
       'Neubau von zwei Häusern', 'Arbeiten am Behördenzentrum',
       'Erweiterung Fernwärme Warnemünde',
       'Erneuerung Straßenbeleuchtung', 'Neubau von Wohneinheiten',
       'Querschläge/Suchschachtungen/Bau provisorische Bushaltestelle',
       'Umverlegung Trinkwasserleitung, SW & RW-Kanal',
       'Gebäudesanierung', 'Neubau Wohn- und Geschäftshaus Teil B',
       'Herstellung Trinkwassergrundstücksanschluss',
       'Betonierungen/ Materiallieferungen',
       'Sanierung TWL und Einflechtung Mischwasserkanal',
       'Reinigung der verschmutzten Fahrbahn', 'Kanalreparaturen',
       'Tiefgründiger Ausbau', 'Herstellug Zufahrt',
       'Herstellung Hausanschluss',
       'Baustelleneinrichtung für Sanierung Kirrchplatz 10',
       'Gehwegsperrung für Hochbau / Umstrukturierung des gesamten Grundstücks Möllner Str.-Plöner Str. 1-6 und 15, 16, 19',
       'Sanierung Geh- und Radweg', 'Reparatur Fernwärmehausanschluss',
       'Neubau Amtsgebäude', 'Sanierung Mischwasserkanal',
       'Modernisierung Mehrfamilienhaus', 'Grundhafter Ausbau',
       'Hochbauarbeiten, Zimmerarbeiten (Dachstuhl, Dachdecker), Erschließung Tiefbauarbeiten Keller, Lieferverkehr',
       'Neubau straßenbegleitender Geh- und Radweg',
       'Erneuerung Hausanschluss Trinkwasser', 'Rohrleitungsbau',
       'LWL-Anbindung -Bohrung-',
       'Fundamentarbeiten Balkonanlage -Betonpumpe & Betonmischer-',
       'Sanierung Gebäude', 'Hochbau Schröderplatz 3 - 4 -Baustr. Az.',
       'Lkw Entladung am 02. & 04.04.2024',
       'Sanierung Trinkwasserleitung 3. BA',
       'Vermessungsarbeiten auf der Brücke', 'Verlegung Erdkabel',
       'LWL- Anbindung in vorhandene Rohrtrasse',
       'Lieferverkehr und Materiallagerung für Hochbau', 'Abrissarbeiten',
       'Arbeiten an der Turnhalle',
       'Neubau Mehrfamilienhaus -Materiallieferung mit Vollsperrung-',
       'Sanierung Wohnhaus', 'Verlegung Breitbandkabel',
       'Neubau Trinkwasserleitung, Neuanschluss Trinkwassergrundstücksanschluss',
       'Ferwärmeerweiterung und Fernwärmehausanschluss', 'Kabelsanierung',
       'Deckenschluss', 'Baustellenausfahrt',
       'Schachtdeckel Havarie Übernahme VRAO Az.',
       'Hochbauarbeiten -Gerüststellung, Verbau, Betonpumpengänge mit Vollsperrung-',
       'Neubau Hausanschluss Fernwärme, Vollsperrung ab 02.04.2024',
       'Neubau Geh- und Radweg',
       'Arbeiten an den Außenanlagen -BE-Fläche-',
       'Neubau Entwässerungsleitachse',
       'Neubau Kita Gänseblümchen -urspr. Az.',
       'Verlegung Fernwärmeleitung',
       'Verlegung Erdkabel - Hausanschluss-',
       'Breitbandanbindung Platz der Freundschaft 13',
       'Hochbau -urspr. Az.', 'Erschließung Wohngebiet Kiefernweg Los 4',
       'Neubau Einfamiliehaus', 'Prüfung und Isolierung Hausanschluss',
       'Sanierung Schachtdeckel', 'Fernwärmehausanschluss', 'Kraneinsatz',
       'Sanierung Regen- und Trinkwasserleitung', 'Störungsbeseitigung',
       'Havarie MW-Kanal - Baustellenzufahrten', 'Neubau Geschäftshaus',
       'Trennung Trinkwassergrundstücksanschluss',
       'Sicherung Baustellenausfahrt',
       'Ertüchtigung Ost-West-Str. zw. Zufahrt A 19 & Gleis 666',
       'Sanierung/Neubau Hausanschluss', 'Neubau FWL',
       'Be- und Entlladung für Umbau des Hotels',
       'Sicherung Baufeld & Baustellenzufahrt',
       'Sperrung eines Teils des Parkplatzes An der Fischerbastion für vorbereitende Maßnahmen (Munitionssondierung/BE-Fläche)',
       'Ersatzneubau Goetheplatzbrücke',
       'Gerüststellung & BE für Fassadensanierung',
       'Haussanierung (Gerüststellung, BE entlang des Hauses und Lieferzone)',
       'Neubau MFH', 'Baustelleneinrichtung, Containerstellung',
       'Sanierung Banhunterweg/Erneuerung Stromwandler',
       'Fassadenarbeiten -Gerüststellung und Baustelleneinrichtung-',
       'Erweiterung Glasfasernetz', 'Errichtung Hotelkomplex',
       'Sanierung Wasser/ Abwasser in versch. Bauphasen',
       'Reparatur Gehweg',
       '2 Einsätze mit Lkw-Kran für Wechsel Mobilfunkantennen',
       'Herstellung Neubau Haus mit Kran',
       'Grundhafte Sanierung in 2 Abschnitten, Herstellung Bushaltestellen',
       'Neubau Hausanschluss Fernwärme', 'Baugrube für Telekom',
       'Neubau Mehrfamilienhaus mit Tiefgarage',
       'Sanierung Trinkwasserleitung', 'Grundhafte Erneuerung',
       'Neubau Ampelanlage', 'Erneuerung TW-Leitung',
       'Anschluss Personenbahnhof',
       'Arbeiten Hausanschluss Trink- und Schmutzwasser',
       'Errichtung einer Kabelbrücke für E-Versorgung Baustelle',
       'Anbindung Breitband Stadtentsorgung HRO',
       'Baustelleneinrichtung / Belieferung für Neubau Goetheplatzbrücke',
       'Baustraße für Neubau Hort Taklerring'], dtype=object)

Diese Liste gibt ein viel komplexeres Bild und listet eine Vielzahl an Typen von Baumaßnahmen. Wenn wir hier die Funktion value_counts() verwenden, so listet diese auch nicht alle Werte auf:

baust.baumassnahme.value_counts()
baumassnahme
Breitbandausbau                                                  12
Erneuerung Straßenbeleuchtung                                     3
Sanierung Trinkwasserleitung                                      3
Verlegung Fernwärmeleitung                                        2
Verlegung Breitbandkabel                                          2
                                                                 ..
Gebäudesanierung                                                  1
Umverlegung Trinkwasserleitung, SW & RW-Kanal                     1
Querschläge/Suchschachtungen/Bau provisorische Bushaltestelle     1
Neubau von Wohneinheiten                                          1
Baustraße für Neubau Hort Taklerring                              1
Name: count, Length: 140, dtype: int64

Bei Datensätzen mit hunderttausenden von Zeilen kann das sehr lange dauern und kein verwertbares Ergebnis liefern. Deshalb ist es meist sinnvoll erstmal zu prüfen wie viele eindeutige Werte existieren. Diese Anzahl können wir mit der Methode .nunique abfragen:

baust.baumassnahme.nunique()
140

Numerische Werte#

Für numerische Variablen sollte es vermieden werden unique() weil diese meist sehr große Ergebnismengen zurück geben, die selten nützlich sind. Stattdessen ist interessante den Wertebereich der Variable zu verstehen (siehe “Normierung”). Dies gibt einen schnellen Überblick darüber, ob die Werte plausibel sind, z.B. ob negative Alterswerte auftreten oder ungewöhnlich große Alterswerte:

baust.longitude.min(), baust.longitude.max()
(12.0120276853359, 12.2248919062465)
baust.latitude.min(), baust.latitude.max()
(54.057580297901, 54.2396815337323)

Der Wertebereich für beide ist für Rostock durchaus plausibel und wir machen uns daher keine Sorgen über zu große oder zu kleine Werte. Siehe Abschnitt “Entfernen unerwünschter Variablen” unten, um fehlende Werte zu erkennen und zu analysieren.

Pandas bietet die Methode describe() an, die es erlaubt diese Statistiken für alle numerische Spalten in einem Aufruf zu extrahieren. Als Ergebnis erhalten wir einen neun Dataframe mit den Statistiken.

baust.describe()
latitude longitude kreis_schluessel gemeindeverband_schluessel gemeinde_schluessel dauer
count 204.000000 204.000000 203.0 203.0 2.030000e+02 204.000000
mean 54.104472 12.110911 13003.0 130030000.0 1.300300e+11 229.857230
std 0.031356 0.034811 0.0 0.0 0.000000e+00 278.425382
min 54.057580 12.012028 13003.0 130030000.0 1.300300e+11 0.166667
25% 54.083985 12.084762 13003.0 130030000.0 1.300300e+11 30.572917
50% 54.090294 12.116659 13003.0 130030000.0 1.300300e+11 97.125000
75% 54.124206 12.136534 13003.0 130030000.0 1.300300e+11 336.750000
max 54.239682 12.224892 13003.0 130030000.0 1.300300e+11 1347.000000

Zählungen und Anteile#

Eine weitere häufige Aufgabe, mit der wir konfrontiert sind, besteht darin, Werte zu zählen oder Anteile von bestimmten Werten zu berechnen. Die Methode .value_counts (siehe oben) kann verwendet werden, um die Anzahl kontinuierlicher Werte zu erhalten. Bei numerischen Werten funktioniert das aufgrund der Vielfalt and Werte nicht gut. Außerdem hilft es auch nicht Fragen die von Bedingungen abhängig sind. Zum Beispiel wollen wir wissen wie viele Baustellen im Jahr 2024 begonnen worden sind. Dafür müssen wir einen logischen Vektor erstellen, der die Bedingung beschreibt, und diesen summieren.

seit2024 = baust.baubeginn > "2024-01-01 00:00:00" # Erzeuge eine logische Variable die True ist wenn der baubeginn nach 2024 ist
seit2024.sum() # Zähle die Baustellen
127

Lassen Sie uns diese Schritte genauer erklären. Zuerst die Zeile

seit2024 = baust.baubeginn > "2024-01-01 00:00:00"
seit2024.head(5)
0     True
1     True
2     True
3     True
4    False
Name: baubeginn, dtype: bool

Erzeuge eine logische Variable die True oder False ist, wenn das Datum des Baubeginns nach 2024 liegt (um genau zu sein machen wir hier einen Stringvergleich und keinen Datumsvergleich. Da wir die Spalte nicht in ein Datum umgewandelt haben.). Ein Beispiel für diese Variable sieht so aus:

Als nächstes addiert seit2024.sum() diese Werte zusammen - denken Sie daran, sobald wir mit logischen Werten rechnen, wird True in “1” und False in “0” umgewandelt, und letzteres spielt beim Addieren keine Rolle. Es ist also so, als würden wir Baustellen zählen. In der Praxis müssen Sie keine separate Variable erstellen, sondern können es direkt berechnen

(baust.baubeginn > "2024-01-01 00:00:00").sum() 
127

Die Berechnung von Proportionen kann auf ähnliche Weise erfolgen. Statt .sum verwenden wir einfach .mean():

(baust.baubeginn > "2024-01-01 00:00:00").mean()
0.6225490196078431

Also gibt es 62% der Baustellen seit 2024.

Filtern von Daten#

Schließlich, lassen wollen wir die Teilmengen der Baustellen erhalten, die entweder im “Hochbau” oder “Straßenbau” liegen. Dies kann mit der Methode .isin() durchgeführt werden, um zu überprüfen, ob ein String in einer gegebenen Liste vorhanden ist:

ab = baust[baust.sparte.isin(["Hochbau", "Straßenbau"])]
ab.head(5)
latitude longitude uuid kreis_name kreis_schluessel gemeindeverband_name gemeindeverband_schluessel gemeinde_name gemeinde_schluessel strasse_name strasse_schluessel sparte von nach baubeginn bauende verkehrsbeeintraechtigungen baumassnahme dauer
5 54.147466 12.065587 0cfabba6-211c-42b0-b10c-3f2c6b2103e0 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Flensburger Str. 01930 Hochbau 25 26 2023-11-22 07:00:00+01:00 2024-04-02 17:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Verkeh... Containerstellung 132.375000
8 54.143917 12.061527 140210b9-601a-482a-a14b-8f13a46879ef Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Möllner Str. 00600 Hochbau NaN NaN 2024-03-25 07:00:00+01:00 2025-12-31 18:00:00+01:00 Sicherungsmaßnahmen entlang der Straße Kennzeichnung Baustellenausfahrten 646.458333
9 54.135178 12.093463 150a9ed8-a7b1-4bf1-8e3f-95e569f480e6 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Schmarl-Dorf 08280 Straßenbau NaN NaN 2024-01-01 00:00:00+01:00 2025-01-01 00:00:00+01:00 Sicherungsmaßnahmen entlang der Straße, Verkeh... Wartungs- und Instandsetzungsarbeiten innerhal... 366.000000
13 54.179967 12.084384 18728d03-7a43-4372-8564-f7519b29a64f Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Luisenstr. 06760 Hochbau 5 NaN 2024-03-22 07:00:00+01:00 2024-03-28 18:00:00+01:00 Sicherungsmaßnahmen entlang der Straße Sanierungsarbeiten 6.458333
14 54.089303 12.108991 1a589275-aebe-497f-b981-1e8f09b81473 Rostock 13003.0 Rostock, Hanse- und Universitätsstadt 130030000.0 Rostock, Hanse- und Universitätsstadt 1.300300e+11 Waldemarstr. 09410 Straßenbau 20 34 2024-03-04 07:00:00+01:00 2024-03-28 18:00:00+01:00 halbseitige Sperrung, Sicherungsmaßnahmen entl... Schutzrohrverlegung 24.458333

Auswahl von Variablen#

Häufig enthalten Datensätze Variablen, die für die aktuelle Analyse irrelevant sind. In einem solchen Fall möchten wir entweder nur die relevanten auswählen (wenn es nur wenige solcher Variablen gibt) oder die irrelevanten entfernen (wenn es nicht zu viele davon gibt). Das Entfernen irrelevanter Informationen hilft uns, die Daten besser zu sehen, sie besser zu verstehen und somit weniger Codierfehler zu haben. Es hilft auch bei bestimmten Transformationen, bei denen wir jetzt möglicherweise auf den gesamten reduzierten Datenrahmen anstatt nur auf ausgewählte Variablen arbeiten möchten. Schließlich hilft es auch, Speicher zu sparen und die Verarbeitungsgeschwindigkeit zu erhöhen.

Auswahl der gewünschten Variablen#

Wir haben besprochen, wie man die gewünschten Variablen auswählt, aber wir werden es hier auch kurz überprüfen. Sie können nur die gewünschten Variablen mit dataframe[["var1", "var2", ..."]] auswählen.

Der Baustellendatensatz enthält zum Beispiel viele Spalten wie “uuid”, “kreis_name”, “kreis_schluessel”, “gemeindeverband_name”, “gemeindeverband_schluessel”, “gemeinde_name”, “gemeinde_schluessel”, “strasse_name”, oder “strasse_schluessel” die redundant sind und nicht gebraucht werden. Deshalb können wir den Datensatz in seiner Komplexität gut reduzieren, indem wir einfach nur die relevanten Spalten auswählen:

baust[["sparte", "baumassnahme", "verkehrsbeeintraechtigungen", "baubeginn", "bauende", "latitude", "longitude", "dauer"]].sample(4)
sparte baumassnahme verkehrsbeeintraechtigungen baubeginn bauende latitude longitude dauer
82 Hochbau Modernisierung Mehrfamilienhaus Sicherungsmaßnahmen entlang der Straße 2023-04-17 07:00:00+02:00 2024-05-03 18:00:00+02:00 54.113939 12.140877 382.458333
15 Straßenbau Herstellung Gehweg und Parkflächen halbseitige Sperrung, Sicherungsmaßnahmen entl... 2024-02-12 07:00:00+01:00 2024-09-30 15:00:00+02:00 54.145707 12.060521 231.291667
202 Hochbau Baustraße für Neubau Hort Taklerring Sicherungsmaßnahmen entlang der Straße, Sicher... 2023-06-01 00:00:00+02:00 2025-08-31 00:00:00+02:00 54.151488 12.080890 822.000000
26 Straßenbau Sanierung Gehweg Sicherungsmaßnahmen entlang der Straße, Sicher... 2024-03-04 07:00:00+01:00 2024-12-20 18:00:00+01:00 54.101617 12.155783 291.458333

Bitte beachten Sie, dass die Spalten in der Reihenfolge zurückgegeben werden, in der sie in der Auswahl aufgeführt sind, nicht in der Originalreihenfolge.

Anstatt den gesamten Datensatz zu laden und später die gewünschten Variablen auszuwählen, können wir auch angeben, welche Spalten mit pd.read_csv gelesen werden sollen.

pd.read_csv("../data/Baustellen/baustellen_flat.csv", 
            usecols=["latitude", "longitude", "sparte", "baubeginn", "bauende", "verkehrsbeeintraechtigungen", "baumassnahme", "dauer"]).sample(4)
latitude longitude sparte baubeginn bauende verkehrsbeeintraechtigungen baumassnahme dauer
4 54.074173 12.123860 Wasserleitung 2023-03-06 00:00:00+01:00 2024-08-01 00:00:00+02:00 Sicherungsmaßnahmen entlang der Straße, Sicher... Sanierung Mischwasser- und Trinkwasserleitung 513.958333
128 54.146542 12.174334 Wasserleitung 2023-12-11 00:00:00+01:00 2024-04-01 00:00:00+02:00 halbseitige Sperrung, Sicherungsmaßnahmen entl... Herstellung Trinkwassergrundstücksanschluss 111.958333
23 54.082812 12.126326 Telefonnetz 2024-03-18 07:00:00+01:00 2024-04-26 18:00:00+02:00 Sicherungsmaßnahmen entlang des Gehwegs Tiefbauarbeiten für Breitbandausbau 39.416667
150 54.087610 12.114437 Gebäudesanierung 2023-11-22 00:00:00+01:00 2024-05-01 00:00:00+02:00 NaN NaN 160.958333

Dies erreicht ein ähnliches Ergebnis, obwohl die Spalten jetzt in der ursprünglichen Reihenfolge und nicht in der angegebenen Reihenfolge sind.

Entfernen unerwünschter Variablen#

Alternativ können wir, wenn wir die meisten Variablen behalten möchten, stattdessen angeben, welche entfernt werden sollen. Dies kann mit der Methode .drop erfolgen. Im Folgenden lassen wir eine große Anzahl von Variablen fallen:

baust.drop(["uuid", "kreis_name", "kreis_schluessel", "gemeindeverband_name", "gemeindeverband_schluessel", "gemeinde_name", "gemeinde_schluessel", "strasse_name", "strasse_schluessel"],
             axis=1 ).sample(4)
latitude longitude sparte von nach baubeginn bauende verkehrsbeeintraechtigungen baumassnahme dauer
53 54.094395 12.084564 Straßenbau Heinrich-Schütz-Str. (03440) Kuphalstr. (06340) 2023-12-13 07:00:00+01:00 2024-05-31 16:00:00+02:00 halbseitige Sperrung, Sicherungsmaßnahmen entl... Querschläge/Suchschachtungen/Bau provisorische... 170.333333
175 54.086582 12.084801 Grünpflege Tschaikowskistr. (09060) Johannes-Kepler-Str. (10650) 2024-03-07 07:00:00+01:00 2024-03-31 15:00:00+02:00 Sicherungsmaßnahmen entlang des Gehwegs, Sperr... Baumpflanzung 24.291667
119 54.174596 12.092916 Kabelnetz 3 6 2024-02-05 07:00:00+01:00 2024-03-28 18:00:00+01:00 NaN Kabelsanierung 52.458333
182 54.077366 12.134501 Straßenbau NaN NaN 2024-01-22 00:00:00+01:00 2026-12-24 00:00:00+01:00 halbseitige Sperrung, Sicherungsmaßnahmen entl... Grundhafte Sanierung in 2 Abschnitten, Herstel... 1067.000000

Beachten Sie, dass wir .drop mitteilen müssen, dass wir Spalten mit diesen Namen zu entfernen und nicht Zeilen mit diesem Index. Dies ist, was das Argument axis=1 bewirkt. Andernfalls erhalten wir einen Fehler:

baust.drop(["uuid", "kreis_name", "kreis_schluessel", "gemeindeverband_name", "gemeindeverband_schluessel", "gemeinde_name", "gemeinde_schluessel", "strasse_name", "strasse_schluessel"]).sample(4)
---------------------------------------------------------------------------
KeyError                                  Traceback (most recent call last)
Cell In[25], line 1
----> 1 baust.drop(["uuid", "kreis_name", "kreis_schluessel", "gemeindeverband_name", "gemeindeverband_schluessel", "gemeinde_name", "gemeinde_schluessel", "strasse_name", "strasse_schluessel"]).sample(4)

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/frame.py:5588, in DataFrame.drop(self, labels, axis, index, columns, level, inplace, errors)
   5440 def drop(
   5441     self,
   5442     labels: IndexLabel | None = None,
   (...)   5449     errors: IgnoreRaise = "raise",
   5450 ) -> DataFrame | None:
   5451     """
   5452     Drop specified labels from rows or columns.
   5453 
   (...)   5586             weight  1.0     0.8
   5587     """
-> 5588     return super().drop(
   5589         labels=labels,
   5590         axis=axis,
   5591         index=index,
   5592         columns=columns,
   5593         level=level,
   5594         inplace=inplace,
   5595         errors=errors,
   5596     )

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/generic.py:4807, in NDFrame.drop(self, labels, axis, index, columns, level, inplace, errors)
   4805 for axis, labels in axes.items():
   4806     if labels is not None:
-> 4807         obj = obj._drop_axis(labels, axis, level=level, errors=errors)
   4809 if inplace:
   4810     self._update_inplace(obj)

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/generic.py:4849, in NDFrame._drop_axis(self, labels, axis, level, errors, only_slice)
   4847         new_axis = axis.drop(labels, level=level, errors=errors)
   4848     else:
-> 4849         new_axis = axis.drop(labels, errors=errors)
   4850     indexer = axis.get_indexer(new_axis)
   4852 # Case for non-unique axis
   4853 else:

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/indexes/base.py:7136, in Index.drop(self, labels, errors)
   7134 if mask.any():
   7135     if errors != "ignore":
-> 7136         raise KeyError(f"{labels[mask].tolist()} not found in axis")
   7137     indexer = indexer[~mask]
   7138 return self.delete(indexer)

KeyError: "['uuid', 'kreis_name', 'kreis_schluessel', 'gemeindeverband_name', 'gemeindeverband_schluessel', 'gemeinde_name', 'gemeinde_schluessel', 'strasse_name', 'strasse_schluessel'] not found in axis"

Insbesondere bedeutet dies, dass .drop einen Fehler verursacht, wenn wir eine nicht vorhandene Spalte entfernen möchten. Dies kann Probleme verursachen, wenn Daten in Notebooks überschrieben werden: Im ersten Durchlauf entfernen Sie eine Variable, und wenn Sie dieselbe Zelle erneut ausführen, ist die Variable weg und Sie erhalten einen Fehler. Wie immer sollten Sie entweder Daten umbenennen, das erneute Ausführen des Datenbereinigungsschritts vermeiden oder Fehlerbehandlung einbeziehen.

Gruppierte Operationen#

Wie andere große DataFrame-Bibliotheken unterstützt auch Pandas gruppierte Operationen. Gruppierte Operationen werden im Wesentlichen wie folgt durchgeführt: Zuerst wird die Daten anhand der Gruppierungsvariablenwerte in Gruppen aufgeteilt. Zweitens werden die folgenden Operationen auf allen Gruppen unabhängig voneinander durchgeführt. Und schließlich werden alle Ergebnisse wieder zusammengeführt, zusammen mit Gruppenindikatoren.

Gruppieren in Pandas kann mit der Methode .groupby durchgeführt werden. Es erwartet eine Liste von Gruppierungsvariablen. Wir demonstrieren dies, indem wir den Baustellendauer nach Sparte berechnen. .groupby erstellt lediglich ein gruppiertes Datenframe, der dann um eine Aggregationsfunktion erweitert wird um einen Ergebnisdataframe zu erhalten.

baust.groupby("sparte").dauer.mean()
sparte
Fernwärmeleitung       75.138889
Gasleitung            185.750000
Gebäudesanierung      179.201786
Gleisbau              939.541667
Grünpflege             68.593750
Hochbau               466.585417
Kabelnetz              51.462121
Kranarbeiten            3.500000
Lichtsignalanlage     260.472222
Straßenbau            295.767857
Straßenbeleuchtung     46.416667
Stromnetz              71.640625
Telefonnetz            54.661458
Wasserleitung         198.724167
privat                  2.250000
Name: dauer, dtype: float64

Das Ergebnis ist eine Serie mit zwei Werten (die numerische dauer) mit dem Index der kategorischen Werte von sparte.

Wir können auch nach mehr als einer Variable gruppieren. In diesem Fall müssen wir diese als Liste angeben, z.B. um die Baustellendauer nach Sparte und Baumaßnahme zu berechnen, können wir das tun.

baust.groupby(["sparte", "baumassnahme"]).dauer.mean()
sparte            baumassnahme                                              
Fernwärmeleitung  Erweiterung Fernwärme Warnemünde                              316.375000
                  FW-Leitung i.a. der Hanseatic                                  39.375000
                  Fernwärme                                                      39.333333
                  Fernwärmeerweiterung                                          186.041667
                  Fernwärmehausanschluss                                         32.000000
                                                                                   ...    
Wasserleitung     Sanierung Wasser/ Abwasser in versch. Bauphasen               494.958333
                  Schachtdeckel Havarie Übernahme VRAO Az.                      190.000000
                  Trennung Trinkwassergrundstücksanschluss                        3.000000
                  Umverlegung Trinkwasserleitung, SW & RW-Kanal                 303.000000
privat            Fundamentarbeiten Balkonanlage -Betonpumpe & Betonmischer-      2.250000
Name: dauer, Length: 145, dtype: float64

Die Methode .groupby unterstützt weitere Optionen, z.B. kann man die Gruppenindikatoren als Variablen behalten anstatt als Index.

Zeichenkettenoperationen#

Pandas öffnet Zeichenkettenvariablen für eine große Liste von Zeichenkettenfunktionen mit dem Attribut .str. Diese replizieren größtenteils das re Modul, aber die Syntax ist anders und oft sind auch die Funktionsnamen unterschiedlich. Wir gehen hier durch eine Reihe von Beispielen, nämlich

  • str.contains zum Suchen von Mustern in Zeichenketten

  • str.match um festzustellen, ob eine Zeichenkette mit einem bestimmten Muster beginnt

  • str.replace um Teile von Zeichenketten zu ersetzen

  • str.split zum Zerteilen von Zeichenketten in Wörter

Viele der Zeichenkettenfunktionen nutzen reguläre Ausdrücke, daher ist es nützlich, eine Vorstellung von den Grundlagen regulärer Ausdrücke zu haben.

Lassen Sie uns Baustellendaten verwenden und analysieren, ob es ein regelmäßiges Muster in den Baumaßnahmen gibt. Wenn wir nur unique aufrufen, erhalten wir eine sehr unübersichtliche Liste.

baust.baumassnahme.unique()[:10]
array(['FW-Leitung i.a. der Hanseatic', 'Baumpflanzung',
       'Einbau Trinkwasserschieber', nan,
       'Sanierung Mischwasser- und Trinkwasserleitung',
       'Containerstellung', 'Kennzeichnung Baustellenausfahrten',
       'Wartungs- und Instandsetzungsarbeiten innerhalb der konzessionierten Strecke',
       'Grünraumarbeiten, Sanierung Außenanlagen',
       'Ausschleifung - Muffung, StS Bussebart'], dtype=object)

Allerdings sehen wir auch, dass einige Wörter wie z.B. “Sanierung” wiederholt vorkommen. Vieleicht ist es sinnvoller, die Zeichenkette in einzelne Worte zu zerlegen und diese zu zählen. Hierfür müssen wir mehrere Methoden verknüpfen, indem wir die Zeichenkette zuerst in Listen an Wörten mit split an allen vorkommenden Leerzeichen und Satzzeichen zerlegen. Mit stack() kombinieren wir diese Liste in eine neue Serie und zählen dann mit value_counts() wie häufig welche Wörter vorkommen.

baust.baumassnahme.str.split(expand=True).stack().value_counts()
Neubau                 25
und                    21
Sanierung              19
Breitbandausbau        13
für                    12
                       ..
straßenbegleitender     1
Trinkwasser             1
LWL-Anbindung           1
-Bohrung-               1
Taklerring              1
Name: count, Length: 283, dtype: int64

Siehe da: Worte wie “Neubau” und “Sanierung” kommen durchaus oft vor.

Finde Muster in Zeichenketten#

Als ersten Schritt wollen wir alle Baumaßnamen identifizieren, die das Wort “Sanierung” enthalten. Muster in Zeichenfolgen können mit str.contains(Muster) identifiziert werden. Diese Funktion gibt einen Vektor von Wahrheitswerten zurück, je nachdem, ob der ursprüngliche Zeichenvektor das Muster enthält:

baust.sparte.str.contains("sanierung", na=False, regex=False, case=False).head(4)
0    False
1    False
2    False
3     True
Name: sparte, dtype: bool

Das Argument na gibt an, was mit fehlenden Werten zu tun ist, hier zwingen wir sie, den Wert false zurückzugeben. regex=False bedeutet, dass das Muster, das wir angeben, kein regulärer Ausdruck ist. Einfache Muster sind einfacher und schneller zu verwenden als reguläre Ausdrücke, aber letztere sind viel leistungsfähiger, aber auch deutlich komplizierter. Der Parameter case gibt an, dass die Funktion nicht auf Groß/Kleinschreibung achten soll.

Jetzt wollen wir uns mal die Baumaßnahmen ansehen, die den Begriff Sanierung enthalten

baust.baumassnahme[baust.baumassnahme.str.contains("sanierung", na=False, regex=False, case=False)].value_counts()
baumassnahme
Sanierung Trinkwasserleitung                                            3
Sanierungsarbeiten                                                      2
Sanierung Trinkwasserleitung 3. BA                                      1
Grundhafte Sanierung in 2 Abschnitten, Herstellung Bushaltestellen      1
Sanierung Wasser/ Abwasser in versch. Bauphasen                         1
Sanierung Banhunterweg/Erneuerung Stromwandler                          1
Haussanierung (Gerüststellung, BE entlang des Hauses und Lieferzone)    1
Gerüststellung & BE für Fassadensanierung                               1
Sanierung/Neubau Hausanschluss                                          1
Sanierung Regen- und Trinkwasserleitung                                 1
Sanierung Schachtdeckel                                                 1
Kabelsanierung                                                          1
Sanierung Wohnhaus                                                      1
Sanierung Mischwasser- und Trinkwasserleitung                           1
Grünraumarbeiten, Sanierung Außenanlagen                                1
Sanierung Mischwasserkanal                                              1
Sanierung Geh- und Radweg                                               1
Baustelleneinrichtung für Sanierung Kirrchplatz 10                      1
Sanierung TWL und Einflechtung Mischwasserkanal                         1
Gebäudesanierung                                                        1
Gehwegsanierung                                                         1
Gehwegsanierung und Verkehrsberuhigung                                  1
Sanierung Altbau                                                        1
Sanierung Gehweg                                                        1
Sanierung Gebäude                                                       1
Name: count, dtype: int64

Bitte beachten Sie: Zuerst müssen Sie das Attribut .str verwenden, um eine Spalte für Zeichenfolgenoperationen zu öffnen. .cabin.contains() funktioniert nicht. .contains gibt einen logischen Wert zurück oder NA im Falle eines fehlenden Eintrags.

Ersetzen von Zeichenfolgen#

Zuletzt wollen wir im Datensatz die deutschen Umlaute ersetzen, z.B. weil diese Probleme mit ML-Modellen bereiten könnten. Dies kann mit der Methode str.replace gemacht werden. Sie hat zwei Argumente: Muster und Ersatz. Wir geben das erstes einen Umlaut wie ä an und ersetzen ihn einfach durch ae:

baust.sparte.str.replace("ä", "ae").replace("ö", "oe").replace("ü", "ue").replace("ß", "ss").value_counts()
sparte
Hochbau               40
Gebaeudesanierung     35
Straßenbau            35
Wasserleitung         25
Fernwaermeleitung     18
Telefonnetz           16
Kabelnetz             11
Stromnetz              8
Grünpflege             4
Lichtsignalanlage      3
Straßenbeleuchtung     3
Gleisbau               2
Kranarbeiten           2
Gasleitung             1
privat                 1
Name: count, dtype: int64

In praktischer Hinsicht ist es oft nützlich, den Originalspalte zu behalten, um zu prüfen, ob die Ersetzung richtig durchgeführt wurde. Dies kann erreicht werden, indem eine neue Spalte im Dataframe erstellt wird und eine Stichprobe der alten und neuen Variablen ausgedruckt wird:

baust["sparte_ASCII"] = baust.sparte.str.replace("ä", "ae").replace("ö", "oe").replace("ü", "ue").replace("ß", "ss")
baust[["sparte", "sparte_ASCII"]].value_counts()
sparte              sparte_ASCII      
Hochbau             Hochbau               40
Gebäudesanierung    Gebaeudesanierung     35
Straßenbau          Straßenbau            35
Wasserleitung       Wasserleitung         25
Fernwärmeleitung    Fernwaermeleitung     18
Telefonnetz         Telefonnetz           16
Kabelnetz           Kabelnetz             11
Stromnetz           Stromnetz              8
Grünpflege          Grünpflege             4
Lichtsignalanlage   Lichtsignalanlage      3
Straßenbeleuchtung  Straßenbeleuchtung     3
Gleisbau            Gleisbau               2
Kranarbeiten        Kranarbeiten           2
Gasleitung          Gasleitung             1
privat              privat                 1
Name: count, dtype: int64

Umgang mit Zeitstempeln#

Wenn man Zeitreihen analysiert, muss man immer mit Zeitstempeln umgehen. Das ist leider fehleranfälliger, als man vermuten mag, weil es sehr viele unterschiedliche Arten gibt Zeitstemmpel darzustellen und es Besonderheiten wie Zeitzonen, Sommer/Winter-Zeit, oder Schaltjahre gibt.

Prinzipiell können wir Datumsangaben und Zeiten mit der Funktion pd.to_datetime in ein richtiges Datetime-Objekt umwandeln, mit dem man auch korrekte Zeitberechnungen machen kann (und nicht solche Workarounds, wie oben). Bei einzelnen Werten kann die Funktion sogar Datums und Zeitformate selbst erkennen.

print(pd.to_datetime("2023-03-08"))
print(pd.to_datetime("03/09/2023"))
print(pd.to_datetime("10-Mar-2023"))
print(pd.to_datetime("2023-03-11"))
2023-03-08 00:00:00
2023-03-09 00:00:00
2023-03-10 00:00:00
2023-03-11 00:00:00
print(pd.to_datetime("12:00"))
print(pd.to_datetime("14:30:00"))
print(pd.to_datetime("10:00:00.400"))
print(pd.to_datetime("18:00+0030"))
2025-08-05 12:00:00
2025-08-05 14:30:00
2025-08-05 10:00:00.400000
2025-08-05 18:00:00+00:30
print(pd.to_datetime("2023-03-08 12:00"))
print(pd.to_datetime("03/09/2023 14:30:00"))
print(pd.to_datetime("10-Mar-2023 10:00:00.400"))
print(pd.to_datetime("2023-03-11T18:00+0030"))
2023-03-08 12:00:00
2023-03-09 14:30:00
2023-03-10 10:00:00.400000
2023-03-11 18:00:00+00:30

Wenn wir diese allerdings in einer Serie kombinieren, funktioniert das nicht mehr

# Beispieldaten
data = pd.DataFrame({
    "Datumzeit": ["2023-03-08 12:00","03/09/2023 14:30:00","10-Mar-2023 10:00:00.400","2023-03-11T18:00+0030"]
})
# Spalte "Uhrzeit" in Zeitformat konvertieren
data["DatumzeitDT"] = pd.to_datetime(data["Datumzeit"])
---------------------------------------------------------------------------
ValueError                                Traceback (most recent call last)
Cell In[38], line 2
      1 # Spalte "Uhrzeit" in Zeitformat konvertieren
----> 2 data["DatumzeitDT"] = pd.to_datetime(data["Datumzeit"])

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/tools/datetimes.py:1072, in to_datetime(arg, errors, dayfirst, yearfirst, utc, format, exact, unit, infer_datetime_format, origin, cache)
   1070         result = arg.map(cache_array)
   1071     else:
-> 1072         values = convert_listlike(arg._values, format)
   1073         result = arg._constructor(values, index=arg.index, name=arg.name)
   1074 elif isinstance(arg, (ABCDataFrame, abc.MutableMapping)):

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/tools/datetimes.py:435, in _convert_listlike_datetimes(arg, format, name, utc, unit, errors, dayfirst, yearfirst, exact)
    433 # `format` could be inferred, or user didn't ask for mixed-format parsing.
    434 if format is not None and format != "mixed":
--> 435     return _array_strptime_with_fallback(arg, name, utc, format, exact, errors)
    437 result, tz_parsed = objects_to_datetime64(
    438     arg,
    439     dayfirst=dayfirst,
   (...)    443     allow_object=True,
    444 )
    446 if tz_parsed is not None:
    447     # We can take a shortcut since the datetime64 numpy array
    448     # is in UTC

File /opt/hostedtoolcache/Python/3.11.0/x64/lib/python3.11/site-packages/pandas/core/tools/datetimes.py:469, in _array_strptime_with_fallback(arg, name, utc, fmt, exact, errors)
    458 def _array_strptime_with_fallback(
    459     arg,
    460     name,
   (...)    464     errors: str,
    465 ) -> Index:
    466     """
    467     Call array_strptime, with fallback behavior depending on 'errors'.
    468     """
--> 469     result, tz_out = array_strptime(arg, fmt, exact=exact, errors=errors, utc=utc)
    470     if tz_out is not None:
    471         unit = np.datetime_data(result.dtype)[0]

File pandas/_libs/tslibs/strptime.pyx:501, in pandas._libs.tslibs.strptime.array_strptime()

File pandas/_libs/tslibs/strptime.pyx:451, in pandas._libs.tslibs.strptime.array_strptime()

File pandas/_libs/tslibs/strptime.pyx:583, in pandas._libs.tslibs.strptime._parse_with_format()

ValueError: time data "03/09/2023 14:30:00" doesn't match format "%Y-%m-%d %H:%M", at position 1. You might want to try:
    - passing `format` if your strings have a consistent format;
    - passing `format='ISO8601'` if your strings are all ISO8601 but not necessarily in exactly the same format;
    - passing `format='mixed'`, and the format will be inferred for each element individually. You might want to use `dayfirst` alongside this.

Deshalb muss man zuerst darauf achten, dass entsprechende Zeitstempel immer in einem festen Format sind.

# Beispieldaten
data = pd.DataFrame({
    "Datumzeit": [
        "2023-03-08T12:00:00.000+0030",
        "2023-03-09T14:30:00.000+0030",
        "2023-03-10T10:00:00.400+0030",
        "2023-03-11T18:00:00.000+0030"]
})
# Spalte "Uhrzeit" in Zeitformat konvertieren
data["DatumzeitDT"] = pd.to_datetime(data["Datumzeit"], format='ISO8601') 
data.DatumzeitDT
0          2023-03-08 12:00:00+00:30
1          2023-03-09 14:30:00+00:30
2   2023-03-10 10:00:00.400000+00:30
3          2023-03-11 18:00:00+00:30
Name: DatumzeitDT, dtype: datetime64[ns, UTC+00:30]

Bei unsicheren Mustern kann man das Format direkt spezifizieren, Dafür wird die strftime-Notation verwendet

# Spalte "Uhrzeit" in Zeitformat konvertieren
data["DatumzeitDT"] = pd.to_datetime(data["Datumzeit"], format="%Y-%m-%dT%H:%M:%S.%f%z")
data.DatumzeitDT
0          2023-03-08 12:00:00+00:30
1          2023-03-09 14:30:00+00:30
2   2023-03-10 10:00:00.400000+00:30
3          2023-03-11 18:00:00+00:30
Name: DatumzeitDT, dtype: datetime64[ns, UTC+00:30]

Mit dem gleichen Muster können Datumsangaben auch wieder in Strings umgewandelt werden.

data.DatumzeitDT.dt.strftime('%Y-%m-%d %H:%M')
0    2023-03-08 12:00
1    2023-03-09 14:30
2    2023-03-10 10:00
3    2023-03-11 18:00
Name: DatumzeitDT, dtype: object