Basics der Versionsverwaltung mit Git (lokale Repos)
Änderungen an Dateien (in der Workingcopy) werden mit git add
zum "Staging" (Index) hinzugefügt.
Dies ist eine Art Sammelbereich für Änderungen, die mit dem nächsten Commit in das Repository
überführt werden. Neue (bisher nicht versionierte Dateien) müssen ebenfalls zunächst mit
git add
zum Staging hinzugefügt werden.
Änderungen kann man mit git log
betrachten, dabei erhält man u.a. eine Liste der Commits und der
jeweiligen Commmit-Messages.
Mit git diff
kann man gezielt Änderungen zwischen Commits oder Branches betrachten.
Mit git tag
kann man bestimmte Commits mit einem "Stempel" (zusätzlicher Name) versehen, um diese
leichter finden zu können.
Wichtig sind die Commit-Messages: Diese sollten eine kurze Zusammenfassung haben, die aktiv formuliert wird (was ändert dieser Commit: "Formatiere den Java-Code entsprechend Style"; nicht aber "Java-Code nach Style formatiert"). Falls der Kommentar länger sein soll, folgt eine Leerzeile auf die erste Zeile (Zusammenfassung) und danach ein Block mit der längeren Erklärung.
- (K3) Umgang mit Dateien: Hinzufügen zum und Löschen aus Repo
- (K3) Umgang mit Änderungen: Hinzufügen zum Staging und Commit
- (K3) Herausfinden von Unterschieden, Ansehen der Historie
- (K3) Ignorieren von Dateien und Ordnern
Versionsverwaltung mit Git: Typische Arbeitsschritte
-
Repository anlegen (oder clonen)
-
Dateien neu erstellen (und löschen, umbenennen, verschieben)
-
Änderungen einpflegen ("committen")
-
Änderungen und Logs betrachten
-
Änderungen rückgängig machen
-
Projektstand markieren ("taggen")
-
Entwicklungszweige anlegen ("branchen")
-
Entwicklungszweige zusammenführen ("mergen")
-
Änderungen verteilen (verteiltes Arbeiten, Workflows)
Dateien unter Versionskontrolle stellen
-
git add .
(odergit add <file>
)=> Stellt alle Dateien (bzw. die Datei
<file>
) im aktuellen Verzeichnis unter Versionskontrolle -
git commit
=> Fügt die Dateien dem Repository hinzu
Abfrage mit git status
Änderungen einpflegen
- Abfrage mit:
git status
- "Staging" von modifizierten Dateien:
git add <file>
- Committen der Änderungen im Stage:
git commit
Anmerkung: Alternativ auch mit git commit -m "Kommentar"
, um das Öffnen
des Editors zu vermeiden ... geht einfach schneller ;)
Das "staging area" stellt eine Art Zwischenebene zwischen Working Copy und Repository dar: Die Änderungen sind temporär "gesichert", aber noch nicht endgültig im Repository eingepflegt ("committed").
Man kann den Stage dazu nutzen, um Änderungen an einzelnen Dateien zu sammeln und diese dann (in einem Commit) gemeinsam einzuchecken.
Man kann den Stage in der Wirkung umgehen, indem man alle in der Working Copy
vorliegenden Änderungen per git commit -a -m "Kommentar"
eincheckt. Der
Schalter "-a
" nimmt alle vorliegenden Änderungen an bereits versionierten
Dateien, fügt diese dem Stage hinzu und führt dann den Commit durch. Das ist
das von SVN bekannte Verhalten. Achtung: Nicht versionierte Dateien bleiben
dabei außen vor!
Letzten Commit ergänzen
-
git commit --amend -m "Eigentlich wollte ich das so sagen"
Wenn keine Änderungen im Stage sind, wird so die letzte Commit-Message geändert.
-
git add <file>; git commit --amend
Damit können vergessene Änderungen an der Datei
<file>
zusätzlich im letzten Commit aufgezeichnet werden.In beiden Fällen ändert sich die Commit-ID!
Weitere Datei-Operationen: hinzufügen, umbenennen, löschen
- Neue (unversionierte) Dateien und Änderungen an versionierten Dateien zum Staging hinzufügen:
git add <file>
- Löschen von Dateien (Repo+Workingcopy):
git rm <file>
- Löschen von Dateien (nur Repo):
git rm --cached <file>
- Verschieben/Umbenennen:
git mv <fileAlt> <fileNeu>
Aus Sicht von Git sind zunächst alle Dateien "untracked", d.h. stehen nicht unter Versionskontrolle.
Mit git add <file>
(und git commit
) werden Dateien in den Index (den
Staging-Bereich, d.h. nach dem Commit letztlich in das Repository) aufgenommen.
Danach stehen sie unter "Beobachtung" (Versionskontrolle). So lange, wie eine
Datei identisch zur Version im Repository ist, gilt sie als unverändert
("unmodified"). Eine Änderung führt entsprechend zum Zustand "modified", und
ein git add <file>
speichert die Änderungen im Stage. Ein Commit überführt
die im Stage vorgemerkte Änderung in das Repo, d.h. die Datei gilt wieder
als "unmodified".
Wenn eine Datei nicht weiter versioniert werden soll, kann sie aus dem Repo
entfernt werden. Dies kann mit git rm <file>
geschehen, wobei die Datei auch
aus der Workingcopy gelöscht wird. Wenn die Datei erhalten bleiben soll, aber
nicht versioniert werden soll (also als "untracked" markiert werden soll), dann
muss sie mit git rm --cached <file>
aus der Versionskontrolle gelöscht werden.
Achtung: Die Datei ist dann nur ab dem aktuellen Commit gelöscht, d.h. frühere
Revisionen enthalten die Datei noch!
Wenn eine Datei umbenannt werden soll, geht das mit git mv <fileAlt> <fileNeu>
.
Letztlich ist dies nur eine Abkürzung für die Folge git rm --cached <fileAlt>
,
manuelles Umbenennen der Datei in der Workingcopy und git add <fileNeu>
.
Commits betrachten
-
Liste aller Commits:
git log
git log -<n>
odergit log --since="3 days ago"
Meldungen eingrenzen ...git log --stat
Statistik ...git log --author="pattern"
Commits eines Autorsgit log <file>
Änderungen einer Datei
-
Inhalt eines Commits:
git show
Änderungen und Logs betrachten
-
git diff [<file>]
Änderungen zwischen Workingcopy und letztem Commit (ohne Stage)
Das "staging area" wird beim Diff von Git behandelt, als wären die dort hinzugefügten Änderungen bereits eingecheckt (genauer: als letzter Commit im aktuellen Branch im Repo vorhanden). D.h. wenn Änderungen in einer Datei mittels
git add <datei>
dem Stage hinzugefügt wurden, zeigtgit diff <datei>
keine Änderungen an! -
git diff commitA commitB
Änderungen zwischen Commits
-
Blame:
git blame <file>
Wer hat was wann gemacht?
Dateien ignorieren: .gitignore
- Nicht alle Dateien gehören ins Repo:
- generierte Dateien:
.class
- temporäre Dateien
- generierte Dateien:
- Datei
.gitignore
anlegen und committen- Wirkt auch für Unterordner
- Inhalt: Reguläre Ausdrücke für zu ignorierende Dateien und Ordner
# Compiled source #
*.class
*.o
*.so
# Packages #
*.zip
# All directories and files in a directory #
bin/**/*
Zeitmaschine
-
Änderungen in Workingcopy rückgängig machen
- Änderungen nicht in Stage:
git checkout <file>
odergit restore <file>
- Änderungen in Stage:
git reset HEAD <file>
odergit restore --staged <file>
=> Hinweise von
git status
beachten! - Änderungen nicht in Stage:
-
Datei aus altem Stand holen:
git checkout <commit> <file>
, odergit restore --source <commit> <file>
-
Commit verwerfen, Geschichte neu:
git revert <commit>
Hinweis: In den neueren Versionen von Git ist der Befehl git restore
hinzugekommen, mit
dem Änderungen rückgängig gemacht werden können. Der bisherige Befehl git checkout
steht
immer noch zur Verfügung und bietet über git restore
hinaus weitere Anwendungsmöglichkeiten.
- Stempel (Tag) vergeben:
git tag <tagname> <commit>
- Tags anzeigen:
git tag
undgit show <tagname>
Wann und wie committen?
Jeder Commit stellt einen Rücksetzpunkt dar!
Typische Regeln:
- Kleinere "Häppchen" einchecken: ein Feature oder Task (das nennt man auch atomic commit: das kleinste Set an Änderungen, die gemeinsam Sinn machen und die ggf. gemeinsam zurückgesetzt werden können)
- Logisch zusammenhängende Änderungen gemeinsam einchecken
- Projekt muss nach Commit compilierbar sein
- Projekt sollte nach Commit lauffähig sein
Ein Commit sollte in sich geschlossen sein, d.h. die kleinste Menge an Änderungen enthalten, die gemeinsam einen Sinn ergeben und die (bei Bedarf) gemeinsam zurückgesetzt oder verschoben werden können. Das nennt man auch atomic commit.
Wenn Sie versuchen, die Änderungen in Ihrem Commit zu beschreiben (siehe nächste Folie "Commit-Messages"), dann werden Sie einen atomic commit mit einem kurzen Satz (natürlich im Imperativ!) beschreiben können. Wenn Sie mehr Text brauchen, haben Sie wahrscheinlich keinen atomic commit mehr vor sich.
Lesen Sie dazu auch How atomic Git commits dramatically increased my productivity - and will increase yours too.
Schreiben von Commit-Messages: WARUM?!
Schauen Sie sich einmal einen Screenshot eines git log --oneline 61e48f0..e2c8076
im Dungeon-CampusMinden/Dungeon an:
Nun stellen Sie sich vor, Sie sind auf der Suche nach Informationen, suchen einen bestimmten Commit oder wollen eine bestimmte Änderung finden ...
Wenn man das genauer analysiert, dann stören bestimmte Dinge:
- Mischung aus Deutsch und Englisch
- "Vor-sich-hin-Murmeln": "Layer system 5"
- Teileweise werden Tags genutzt wie
[BUG]
, aber nicht durchgängig - Mischung zwischen verschiedenen Formen: "Repo umbenennen", "Benenne Repo um", "Repo umbenannt"
- Unterschiedliche Groß- und Kleinschreibung
- Sehr unterschiedlich lange Zeilen/Kommentare
Das Beachten einheitlicher Regeln ist enorm wichtig!
Leider sagt sich das so leicht - in der Praxis macht man es dann
doch schnell wieder unsauber. Dennoch, auch im Dungeon-Repo gibt
es einen positiven Trend (git log --oneline 8039d6c..7f49e89
):
Typische Regeln und Konventionen tauchen überall auf, beispielsweise in [Chacon2014] oder bei Tim Pope (siehe nächstes Beispiel) oder bei "How to Write a Git Commit Message".
Short (50 chars or less) summary of changes
More detailed explanatory text, if necessary. Wrap it to about
72 characters or so. In some contexts, the first line is treated
as the subject of an email and the rest of the text as the body.
The blank line separating the summary from the body is critical
(unless you omit the body entirely); tools like rebase can get
confused if you run the two together.
Further paragraphs come after blank lines.
- Bullet points are okay, too
- Typically a hyphen or asterisk is used for the bullet, preceded
by a single space, with blank lines in between, but conventions
vary here
Quelle: "A Note About Git Commit Messages" by Tim Pope on tbaggery.com
Denken Sie sich die Commit-Message als E-Mail an einen zukünftigen Entwickler, der das in fünf Jahren liest!
Vom Aufbau her hat eine E-Mail auch eine Summary und dann den eigentlichen Inhalt ... Erklären Sie das "WARUM" der Änderung! (Das "WER", "WAS", "WANN" wird bereits automatisch von Git aufgezeichnet ...)
Lesen (und beachten) Sie unbedingt auch "How to Write a Git Commit Message"!
Wrap-Up
- Änderungen einpflegen zweistufig (
add
,commit
) - Status der Workingcopy mit
status
ansehen - Logmeldungen mit
log
ansehen - Änderungen auf einem File mit
diff
bzw.blame
ansehen - Projektstand markieren mit
tag
- Ignorieren von Dateien/Ordnern: Datei
.gitignore
Versionierung 101
- Legen Sie ein Repository an.
- Fügen Sie Dateien dem Verzeichnis hinzu und stellen Sie einige davon unter Versionskontrolle.
- Ändern Sie eine Datei und versionieren Sie die Änderung.
- Was ist der Unterschied zwischen "
git add .; git commit
" und "git commit -a
"? - Wie finden Sie heraus, welche Dateien geändert wurden?
- Entfernen Sie eine Datei aus der Versionskontrolle, aber nicht aus dem Verzeichnis!
- Entfernen Sie eine Datei komplett (Versionskontrolle und Verzeichnis).
- Ändern Sie eine Datei und betrachten die Unterschiede zum letzten Commit.
- Fügen Sie eine geänderte Datei zum Index hinzu. Was erhalten Sie bei
git diff <datei>
? - Wie können Sie einen früheren Stand einer Datei wiederherstellen? Wie finden Sie überhaupt den Stand?
- Legen Sie sich ein Java-Projekt in Ihrer IDE an an. Stellen Sie dieses Projekt unter Git-Versionskontrolle. Führen Sie die vorigen Schritte mit Ihrer IDE durch.
Interaktive Git-Tutorials: Schaffen Sie die Rätsel?
- [AtlassianGit] Become a git guru.
Atlassian Pty Ltd, 2022. - [Chacon2014] Pro Git
Chacon, S. und Straub, B., Apress, 2014. ISBN 978-1-4842-0077-3.
Kapitel 2 - [GitCheatSheet] Git Cheat Sheets
Github Inc., 2022.