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.
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 vonbaust
. Hier bedeutet es, das die Spaltesparte
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 Seriesparte
), 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 Zeichenkettenstr.match
um festzustellen, ob eine Zeichenkette mit einem bestimmten Muster beginntstr.replace
um Teile von Zeichenketten zu ersetzenstr.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