-
Notifications
You must be signed in to change notification settings - Fork 312
/
index-en.html
516 lines (409 loc) · 19.9 KB
/
index-en.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
<!DOCTYPE html>
<html lang="en-US">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>A Visual Git Reference</title>
<link rel='stylesheet' type='text/css' href='visual-git-guide.css'>
<script type="text/javascript" src='visual-git-guide.js'></script>
<script data-goatcounter="https://marklodato.goatcounter.com/count"
async src="goatcounter.js"></script>
</head>
<body onload="replace_all_PNGs();">
<h1 id="top">A Visual Git Reference</h1>
<div id="language-box">
<a>Other Languages:</a>
<ul>
<li><a href='index-de.html'>Deutsch</a></li>
<li class="selected">English</li>
<li><a href='index-es.html'>Español</a></li>
<li><a href='index-fr.html'>Français</a></li>
<li><a href='index-it.html'>Italiano</a></li>
<li><a href='index-ja.html'>日本語</a></li>
<li><a href='index-ko.html'>한국어</a></li>
<li><a href='index-pl.html'>Polski</a></li>
<li><a href='index-pt.html'>Português</a></li>
<li><a href='index-ru.html'>Русский</a></li>
<li><a href='index-sk.html'>Slovenčina</a></li>
<li><a href='index-vi.html'>Tiếng Việt</a></li>
<li><a href='index-zh-cn.html'>简体中文</a></li>
<li><a href='index-zh-tw.html'>正體中文</a></li>
</ul>
</div>
<p id="link-to-png">If the images do not work, you can try the <a
href="?no-svg">Non-SVG</a> version of this page.</p>
<p id="link-to-svg">SVG images have been disabled.
<a href="index.html">(Re-enable SVG)</a></p>
<p>This page gives brief, visual reference for the most common commands
in git. Once you know a bit about how git works, this site may solidify
your understanding. If you're interested in how this site was created, see
my <a href='http://github.com/MarkLodato/visual-git-guide'>GitHub
repository</a>.</p>
<p id="link-to-d3">Also recommended: <a
href="http://onlywei.github.io/explain-git-with-d3/#">Visualizing Git
Concepts with D3</a></p>
<h2 id="contents">Contents</h2>
<ol>
<li><a href="#basic-usage">Basic Usage</a></li>
<li><a href="#conventions">Conventions</a></li>
<li><a href="#commands-in-detail">Commands in Detail</a>
<ol>
<li><a href="#diff">Diff</a></li>
<li><a href="#commit">Commit</a></li>
<li><a href="#checkout">Checkout</a></li>
<li><a href="#detached">Committing with a Detached HEAD</a></li>
<li><a href="#reset">Reset</a></li>
<li><a href="#merge">Merge</a></li>
<li><a href="#cherry-pick">Cherry Pick</a></li>
<li><a href="#rebase">Rebase</a></li>
</ol>
</li>
<li><a href="#technical-notes">Technical Notes</a></li>
<li><a href="#appendix-stage">Walkthrough: Watching the effect of
commands</a></li>
</ol>
<h2 id="basic-usage">Basic Usage</h2>
<div class="center"><img src='basic-usage.svg.png'></div>
<p>The four commands above copy files between the working directory, the
stage (also called the index), and the history (in the form of commits).</p>
<ul>
<li><code>git add <em>files</em></code> copies <em>files</em> (at their
current state) to the stage.</li>
<li><code>git commit</code> saves a snapshot of the stage as a
commit.</li>
<li><code>git reset -- <em>files</em></code> unstages files; that is, it
copies <em>files</em> from the latest commit to the stage. Use this
command to "undo" a <code>git add <em>files</em></code>. You can also
<code>git reset</code> to unstage everything.</li>
<li><code>git checkout -- <em>files</em></code> copies <em>files</em> from
the stage to the working directory. Use this to throw away local
changes.</li>
</ul>
<p>You can use <code>git reset -p</code>, <code>git checkout -p</code>, or
<code>git add -p</code> instead of (or in addition to) specifying particular
files to interactively choose which hunks copy.</p>
<p>It is also possible to jump over the stage and check out files directly
from the history or commit files without staging first.</p>
<div class="center"><img src='basic-usage-2.svg.png'></div>
<ul>
<li><code>git commit -a </code> is equivalent to running <tt>git add</tt>
on all filenames that existed in the latest commit, and then running
<tt>git commit</tt>.</li>
<li><code>git commit <em>files</em></code> creates a new commit containing
the contents of the latest commit, plus a snapshot of <em>files</em> taken
from the working directory. Additionally, <em>files</em> are copied to
the stage.</li>
<li><code>git checkout HEAD -- <em>files</em></code> copies <em>files</em>
from the latest commit to both the stage and the working directory.</li>
</ul>
<h2 id="conventions">Conventions</h2>
<p>In the rest of this document, we will use graphs of the following
form.</p>
<div class="center"><img src='conventions.svg.png'></div>
<p>Commits are shown in green as 5-character IDs, and they point to their
parents. Branches are shown in orange, and they point to particular
commits. The current branch is identified by the special reference
<em>HEAD</em>, which is "attached" to that branch. In this image, the five
latest commits are shown, with <em>ed489</em> being the most recent.
<em>main</em> (the current branch) points to this commit, while
<em>stable</em> (another branch) points to an ancestor of <em>main</em>'s
commit.</p>
<h2 id="commands-in-detail">Commands in Detail</h2>
<h3 id="diff">Diff</h3>
<p>There are various ways to look at differences between commits. Below are
some common examples. Any of these commands can optionally take extra
filename arguments that limit the differences to the named files.</p>
<div class="center"><img src='diff.svg.png'></div>
<h3 id="commit">Commit</h3>
<p>When you commit, git creates a new commit object using the files from the
stage and sets the parent to the current commit. It then points the current
branch to this new commit. In the image below, the current branch is
<em>main</em>. Before the command was run, <em>main</em> pointed to
<em>ed489</em>. Afterward, a new commit, <em>f0cec</em>, was created, with
parent <em>ed489</em>, and then <em>main</em> was moved to the new
commit.</p>
<div class="center"><img src='commit-main.svg.png'></div>
<p>This same process happens even when the current branch is an ancestor of
another. Below, a commit occurs on branch <em>stable</em>, which was an
ancestor of <em>main</em>, resulting in <em>1800b</em>. Afterward,
<em>stable</em> is no longer an ancestor of <em>main</em>. To join the two
histories, a <a href='#merge'>merge</a> (or <a href='#rebase'>rebase</a>)
will be necessary.</p>
<div class="center"><img src='commit-stable.svg.png'></div>
<p>Sometimes a mistake is made in a commit, but this is easy to correct with
<code>git commit --amend</code>. When you use this command, git creates a
new commit with the same parent as the current commit. (The old commit will
be discarded if nothing else references it.)</p>
<div class="center"><img src='commit-amend.svg.png'></div>
<p>A fourth case is committing with a <a href="#detached">detached
HEAD</a>, as explained later.</p>
<h3 id="checkout">Checkout</h3>
<p>The checkout command is used to copy files from the history (or stage) to
the working directory, and to optionally switch branches.</p>
<p>When a filename (and/or <code>-p</code>) is given, git copies those files
from the given commit to the stage and the working directory. For example,
<code>git checkout HEAD~ foo.c</code> copies the file <code>foo.c</code>
from the commit called <em>HEAD~</em> (the parent of the current commit) to
the working directory, and also stages it. (If no commit name is given,
files are copied from the stage.) Note that the current branch is not
changed.</p>
<div class="center"><img src='checkout-files.svg.png'></div>
<p>When a filename is <em>not</em> given but the reference is a (local)
branch, <em>HEAD</em> is moved to that branch (that is, we "switch to" that
branch), and then the stage and working directory are set to match the
contents of that commit. Any file that exists in the new commit
(<em>a47c3</em> below) is copied; any file that exists in the old commit
(<em>ed489</em>) but not in the new one is deleted; and any file that exists
in neither is ignored.</p>
<div class="center"><img src='checkout-branch.svg.png'></div>
<p>When a filename is <em>not</em> given and the reference is <em>not</em> a
(local) branch — say, it is a tag, a remote branch, a SHA-1 ID, or
something like <em>main~3</em> — we get an anonymous branch, called
a <em>detached HEAD</em>. This is useful for jumping around the history.
Say you want to compile version 1.6.6.1 of git. You can <code>git checkout
v1.6.6.1</code> (which is a tag, not a branch), compile, install, and then
switch back to another branch, say <code>git checkout main</code>.
However, committing works slightly differently with a detached HEAD; this is
covered <a href="#detached">below</a>.</p>
<div class="center"><img src='checkout-detached.svg.png'></div>
<h3 id="detached">Committing with a Detached HEAD</h3>
<p>When <em>HEAD</em> is detached, commits work like normal, except no named
branch gets updated. (You can think of this as an anonymous branch.)</p>
<div class="center"><img src='commit-detached.svg.png'></div>
<p>Once you check out something else, say <em>main</em>, the commit is
(presumably) no longer referenced by anything else, and gets lost. Note
that after the command, there is nothing referencing <em>2eecb</em>.</p>
<div class="center"><img src='checkout-after-detached.svg.png'></div>
<p>If, on the other hand, you want to save this state, you can create a new
named branch using <code>git checkout -b <em>name</em></code>.</p>
<div class="center"><img src='checkout-b-detached.svg.png'></div>
<h3 id="reset">Reset</h3>
<p>The reset command moves the current branch to another position, and
optionally updates the stage and the working directory. It also is used to
copy files from the history to the stage without touching the working
directory.</p>
<p>If a commit is given with no filenames, the current branch is moved to
that commit, and then the stage is updated to match this commit. If
<code>--hard</code> is given, the working directory is also updated. If
<code>--soft</code> is given, neither is updated.</p>
<div class="center"><img src='reset-commit.svg.png'></div>
<p>If a commit is not given, it defaults to <em>HEAD</em>. In this case,
the branch is not moved, but the stage (and optionally the working
directory, if <code>--hard</code> is given) are reset to the contents of the
last commit.</p>
<div class="center"><img src='reset.svg.png'></div>
<p>If a filename (and/or <code>-p</code>) is given, then the command works
similarly to <a href='#checkout'>checkout</a> with a filename, except only
the stage (and not the working directory) is updated. (You may also specify
the commit from which to take files, rather than <em>HEAD</em>.)</p>
<div class="center"><img src='reset-files.svg.png'></div>
<h3 id="merge">Merge</h3>
<p>A merge creates a new commit that incorporates changes from other
commits. Before merging, the stage must match the current commit. The
trivial case is if the other commit is an ancestor of the current commit, in
which case nothing is done. The next most simple is if the current commit
is an ancestor of the other commit. This results in a <em>fast-forward</em>
merge. The reference is simply moved, and then the new commit is checked
out.</p>
<div class="center"><img src='merge-ff.svg.png'></div>
<p>Otherwise, a "real" merge must occur. You can choose other strategies,
but the default is to perform a "recursive" merge, which basically takes the
current commit (<em>ed489</em> below), the other commit (<em>33104</em>),
and their common ancestor (<em>b325c</em>), and performs a <a
href='http://en.wikipedia.org/wiki/Three-way_merge'>three-way merge</a>.
The result is saved to the working directory and the stage, and then a
commit occurs, with an extra parent (<em>33104</em>) for the new commit.
</p>
<div class="center"><img src='merge.svg.png'></div>
<h3 id="cherry-pick">Cherry Pick</h3>
<p>The cherry-pick command "copies" a commit, creating a new commit on the
current branch with the same message and patch as another commit.</p>
<div class="center"><img src='cherry-pick.svg.png'></div>
<h3 id="rebase">Rebase</h3>
<p>A rebase is an alternative to a <a href='#merge'>merge</a> for combining
multiple branches. Whereas a merge creates a single commit with two
parents, leaving a non-linear history, a rebase replays the commits from the
current branch onto another, leaving a linear history. In essence, this is
an automated way of performing several <a
href='#cherry-pick'>cherry-pick</a>s in a row.</p>
<div class="center"><img src='rebase.svg.png'></div>
<p>The above command takes all the commits that exist in <em>topic</em> but
not in <em>main</em> (namely <em>169a6</em> and <em>2c33a</em>), replays
them onto <em>main</em>, and then moves the branch head to the new tip.
Note that the old commits will be garbage collected if they are no longer
referenced.</p>
<p>To limit how far back to go, use the <code>--onto</code> option. The
following command replays onto <em>main</em> the most recent commits on
the current branch since <em>169a6</em> (exclusive), namely
<em>2c33a</em>.</p>
<div class="center"><img src='rebase-onto.svg.png'></div>
<p>There is also <code>git rebase --interactive</code>, which allows one to
do more complicated things than simply replaying commits, namely dropping,
reordering, modifying, and squashing commits. There is no obvious picture
to draw for this; see <a
href='http://www.kernel.org/pub/software/scm/git/docs/git-rebase.html#_interactive_mode'>git-rebase(1)</a>
for more details.</p>
<h2 id="technical-notes">Technical Notes</h2>
<p>The contents of files are not actually stored in the index
(<em>.git/index</em>) or in commit objects. Rather, each file is stored in
the object database (<em>.git/objects</em>) as a <em>blob</em>, identified
by its SHA-1 hash. The index file lists the filenames along with the
identifier of the associated blob, as well as some other data. For commits,
there is an additional data type, a <em>tree</em>, also identified by its
hash. Trees correspond to directories in the working directory, and contain
a list of trees and blobs corresponding to each filename within that
directory. Each commit stores the identifier of its top-level tree, which
in turn contains all of the blobs and other trees associated with that
commit.</p>
<p>If you make a commit using a detached HEAD, the last commit really is
referenced by something: the reflog for HEAD. However, this will expire
after a while, so the commit will eventually be garbage collected, similar
to commits discarded with <code>git commit --amend</code> or <code>git
rebase</code>.</p>
<h2 id="appendix-stage">Walkthrough: Watching the effect of commands</h2>
<p>The following walks you through changes to a repository so you can see the
effect of the command in real time, similar to how <a
href="http://onlywei.github.io/explain-git-with-d3/#">Visualizing Git
Concepts with D3</a> simulates them visually. Hopefully you find this
useful.</p>
<p>Start by creating some repository:</p>
<pre><code>$ <strong>git init foo</strong>
$ <strong>cd foo</strong>
$ <strong>echo 1 > myfile</strong>
$ <strong>git add myfile</strong>
$ <strong>git commit -m "version 1"</strong>
</code></pre>
<p>Now, define the following functions to help us show information:</p>
<pre><code>show_status() {
echo "HEAD: $(git cat-file -p HEAD:myfile)"
echo "Stage: $(git cat-file -p :myfile)"
echo "Worktree: $(cat myfile)"
}
initial_setup() {
echo 3 > myfile
git add myfile
echo 4 > myfile
show_status
}
</code></pre>
<p>Initially, everything is at version 1.</p>
<pre><code>$ <strong>show_status</strong>
HEAD: 1
Stage: 1
Worktree: 1
</code></pre>
<p>We can watch the state change as we add and commit.</p>
<pre><code>$ <strong>echo 2 > myfile</strong>
$ <strong>show_status</strong>
HEAD: 1
Stage: 1
Worktree: 2
$ <strong>git add myfile</strong>
$ <strong>show_status</strong>
HEAD: 1
Stage: 2
Worktree: 2
$ <strong>git commit -m "version 2"</strong>
[main 4156116] version 2
1 file changed, 1 insertion(+), 1 deletion(-)
$ <strong>show_status</strong>
HEAD: 2
Stage: 2
Worktree: 2
</code></pre>
<p>Now, let's create an initial state where the three are all different.</p>
<pre><code>$ <strong>initial_setup</strong>
HEAD: 2
Stage: 3
Worktree: 4
</code></pre>
<p>Let's watch what each command does. You will see that they match the
diagrams above.</p>
<p><code>git reset -- myfile</code> copies from HEAD to stage:</p>
<pre><code>$ <strong>initial_setup</strong>
HEAD: 2
Stage: 3
Worktree: 4
$ <strong>git reset -- myfile</strong>
Unstaged changes after reset:
M myfile
$ <strong>show_status</strong>
HEAD: 2
Stage: 2
Worktree: 4
</code></pre>
<p><code>git checkout -- myfile</code> copies from stage to worktree:</p>
<pre><code>$ <strong>initial_setup</strong>
HEAD: 2
Stage: 3
Worktree: 4
$ <strong>git checkout -- myfile</strong>
$ <strong>show_status</strong>
HEAD: 2
Stage: 3
Worktree: 3
</code></pre>
<p><code>git checkout HEAD -- myfile</code> copies from HEAD to both stage and
worktree:</p>
<pre><code>$ <strong>initial_setup</strong>
HEAD: 2
Stage: 3
Worktree: 4
$ <strong>git checkout HEAD -- myfile</strong>
$ <strong>show_status</strong>
HEAD: 2
Stage: 2
Worktree: 2
</code></pre>
<p><code>git commit myfile</code> copies from worktree to both stage and
HEAD:</p>
<pre><code>$ <strong>initial_setup</strong>
HEAD: 2
Stage: 3
Worktree: 4
$ <strong>git commit myfile -m "version 4"</strong>
[main 679ff51] version 4
1 file changed, 1 insertion(+), 1 deletion(-)
$ <strong>show_status</strong>
HEAD: 4
Stage: 4
Worktree: 4
</code></pre>
<hr>
<p>Copyright © 2010,
<a href='mailto:[email protected]'>Mark Lodato</a>.
Japanese translation © 2010,
<a href='http://github.com/kazu-yamamoto'>Kazu Yamamoto</a>.
Korean translation © 2011,
<a href='mailto:[email protected]'>Sean Lee</a>.
Russian translation © 2012,
<a href='mailto:[email protected]'>Alex Sychev</a>.
French translation © 2012,
<a href='mailto:[email protected]'>Michel Lefranc</a>.
Chinese translation © 2012,
<a href='mailto:[email protected]'>wych</a>.
Spanish translation © 2012,
<a href='http://www.delucas.com.ar'>Lucas Videla</a>.
Italian translation © 2012,
<a href="mailto:[email protected]">Daniel Londero</a>.
German translation © 2013,
<a href="mailto:[email protected]">Martin Funk</a>.
Slovak translation © 2013,
<a href='https://github.com/llucenic'>Ľudovít Lučenič</a>.
Portuguese translation © 2014,
<a href='mailto:[email protected]'>Gustavo de Oliveira</a>.
Traditional Chinese translation © 2015,
<a href='https://github.com/PeterDaveHello'>Peter Dave Hello</a>.
Polish translation © 2017,
<a href="mailto:[email protected]">Emil Wypych</a>.
</p>
<p><a rel="license"
href="https://creativecommons.org/licenses/by-nc-sa/3.0/us/"><img alt=""
src="https://i.creativecommons.org/l/by-nc-sa/3.0/us/80x15.png"></a>
This work is licensed under a <a rel="license"
href="https://creativecommons.org/licenses/by-nc-sa/3.0/us/">Creative
Commons Attribution-Noncommercial-Share Alike 3.0 United States
License</a>.</p>
<p><a href='translate-en.html'>Want to translate into another
language?</a></p>
</body>
</html>