data:image/s3,"s3://crabby-images/5fad5/5fad50ff97411ff60a197efa651496cb5136f78d" alt=""
How Git's 3-way Merge Algorithm Works
How Git's 3-way Merge Algorithm Works 관련
Get back to the state before applying this patch:
git reset --hard
You have now three versions: the merge base, which is "Commit 10", Paul's branch, and John's branch. In general terms, we can say these are the merge base
, commit A
and commit B
. Notice that the merge base
is by definition an ancestor of both commit A
and commit B
.
To perform the merge, Git looks at the diff between the three different versions of the file in question on these three revisions. In your case, it's the file everyone.md
, and the revisions are "Commit 10", Paul's branch – that is, "Commit 11", and John's branch, that is, "Commit 12".
Git makes the merging decision based on the status of each line in each of these versions.
data:image/s3,"s3://crabby-images/16622/16622b25444c63d2f7259cca2f3fe31cb15241d1" alt="The three versions considered for the 3-way merge<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
In case not all three versions match, that is a conflict. Git can resolve many of these conflicts automatically, as we will now see.
Let's consider specific lines.
The first lines here exist only on Paul's branch:
data:image/s3,"s3://crabby-images/74fa2/74fa240a2f7b0ba663b8c85b2e70b51f0fc3ae89" alt="Lines that appear on Paul's branch only<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
This means that the state of John's branch is equal to the state of the merge base. So the 3-way merge goes with Paul's version.
In general, if the state of the merge base is the same as A
, the algorithm goes with B
. The reason is that since the merge base is the ancestor of both A
and B
, Git assumes that this line hasn't changed in A
, and it has changed in B
, which is the most recent version for that line, and should thus be taken into account.
data:image/s3,"s3://crabby-images/bd20c/bd20c632cca67a3031649c4d9554d92496eff66b" alt="If the state of the merge base is the same as , and this state is different from , the algorithm goes with <br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
A
, and this state is different from B
, the algorithm goes with B
<Source: Brief>
Next, you can see lines where all three versions agree – they exist on the merge base, A
and B
, with equal data.
data:image/s3,"s3://crabby-images/cb5f4/cb5f4c4ac366632e3affcbd6ece301d8a11e5085" alt="Lines where all three versions agree<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
So the algorithm has a trivial choice – just take that version.
data:image/s3,"s3://crabby-images/df509/df5091fc63355c2fa205e6a07f74d4abee87fc8c" alt="In case all three versions agree, the algorithm goes with that single version<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
In a previous example, we saw that if the merge base and A
agree, and B
's version is different, the algorithm picks B
. This works in the other direction too – for example, here you have a line that exists on John's branch, different than that on the merge base and Paul's branch.
data:image/s3,"s3://crabby-images/abe1d/abe1d4af41e95fae141f07ea30caf085f5d35085" alt="A line where Paul's version matches the merge base's version, and John has a different version<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
Hence, John's version is chosen.
data:image/s3,"s3://crabby-images/056fb/056fbd214e8a4c542e768443db8a1830d9d59788" alt="If the state of the merge base is the same as , and this state is different from , the algorithm goes with <br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
B
, and this state is different from A
, the algorithm goes with A
<Source: Brief>
Now consider another case, where both A
and B
agree on a line, but the value they agree upon is different from the merge base
– both John and Paul agreed to change the line "Everyone put their feet down" to "Everyone put their foot down":
data:image/s3,"s3://crabby-images/886eb/886eb39ac8a4e9a12a3aaf7b9bf05ba58e440103" alt="A line where Paul's version matches the John's version; yet the merge base has a different version<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
In this case, the algorithm picks the version on both A
and B
.
data:image/s3,"s3://crabby-images/e37eb/e37eb8d89e231be9aa193b339c8b4d550ffe6acf" alt="In case and agree on a version which is different from the merge base's version, the algorithm picks the version on both and <br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
A
and B
agree on a version which is different from the merge base's version, the algorithm picks the version on both A
and B
<Source: Brief>
Notice this is not a democratic vote. In the previous case, the algorithm picked the minority version, as it resembled the newest version of this line. In this case, it happens to pick the majority – but only because A
and B
are the revisions that agree on the new version.
The same would happen if we used git merge
:
git merge john_branch_3
Without specifying any flags, git merge
will default to using a 3-way merge.
data:image/s3,"s3://crabby-images/80622/80622cda7e7ab72f45b00ae5e94e531be9f305a2" alt="By default, uses a 3-way merge algorithm<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
git merge
uses a 3-way merge algorithm<Source: Brief>
The status of everyone.md
after running the command above would be the same as the result you achieved by applying the patches with git apply -3
.
If you consider the history:
data:image/s3,"s3://crabby-images/76695/76695136b1ec0a79195a1e635e3ad14cd987f352" alt="Git's history after performing the merge<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
You will see that the merge commit indeed has two parents: the first is "Commit 11", that is, where paul_branch_3
pointed to before the merge. The second is "Commit 12", where john_branch_3
pointed to, and still points to now.
What will happen if you now merge from main
? That is, switch to the main branch, which is pointing to "Commit 10":
git checkout main
And then merge Paul's branch?
git merge paul_branch_3
Indeed, a fast forward, as before running this command, main
was an ancestor of paul_branch_3
.
data:image/s3,"s3://crabby-images/63c52/63c529324537ed393a7e3db777e1f7ef5c8d8a8c" alt="A fast-forward merge<br/><Source: <FontIcon icon="fa-brands fa-youtube"/>Brief>"
<Source: Brief>
So, this is a 3-way merge. In general, if all versions agree on a line, then this line is used. If A
and the merge base
match, and B
has another version, B
is taken. In the opposite case, where the merge base
and B
match, the A
version is selected. If A
and B
match, this version is taken, whether the merge base agrees or not.
This description leaves one open question though: What happens in cases where all three versions disagree?
Well, that's a conflict that Git does not resolve automatically. In these cases, Git calls for a human's help.