What is the difference between merge --squash and rebase?

221,344

Solution 1

Both git merge --squash and git rebase --interactive can produce a "squashed" commit. But they serve different purposes.

will produce a squashed commit on the destination branch, without marking any merge relationship. (Note: it does not produce a commit right away: you need an additional git commit -m "squash branch")

This is useful if you want to throw away the source branch completely, going from (schema taken from SO question):

git checkout stable

          X               stable
         /
a---b---c---d---e---f---g tmp

to:

git merge --squash tmp
git commit -m "squash tmp"


# In the following graph, G is c--d--e--f--g squashed together

          X-------------G stable
         /
a---b---c---d---e---f---g tmp

and then deleting tmp branch.


Note: git merge has a --commit option, but it cannot be used with --squash. It was never possible to use --commit and --squash together. Since Git 2.22.1 (Q3 2019), this incompatibility is made explicit:

See commit 1d14d0c (24 May 2019) by Vishal Verma (reloadbrain). (Merged by Junio C Hamano -- gitster -- in commit 33f2790, 25 Jul 2019)

merge: refuse --commit with --squash

Previously, when --squash was supplied, 'option_commit' was silently dropped. This could have been surprising to a user who tried to override the no-commit behavior of squash using --commit explicitly.

git/git builtin/merge.c#cmd_merge() now includes:

if (option_commit > 0)
    die(_("You cannot combine --squash with --commit."));

replays some or all of your commits on a new base, allowing you to squash (or more recently "fix up", see this SO question), going directly to:

git checkout tmp
git rebase -i stable

   stable
      X----------------G tmp
     /
a---b

If you choose to squash all commits of tmp (but, contrary to merge --squash, you can choose to replay some, and squashing others).

So the differences are:

  • squash does not touch your source branch (tmp here) and creates a single commit where you want.
  • rebase allows you to go on on the same source branch (still tmp) with:
    • a new base
    • a cleaner history

Solution 2

Merge commits: retains all of the commits in your branch and interleaves them with commits on the base branchenter image description here

Merge Squash: retains the changes but omits the individual commits from history enter image description here

Rebase: This moves the entire feature branch to begin on the tip of the master branch, effectively incorporating all of the new commits in master

enter image description here

More on here


The first two diagrams come from About pull request merges on the GitHub Docs

Solution 3

Let's start by the following example:

enter image description here

Now we have 3 options to merge changes of feature branch into master branch:

  1. Merge commits
    Will keep all commits history of the feature branch and move them into the master branch
    Will add extra dummy commit.

  2. Rebase and merge
    Will append all commits history of the feature branch in the front of the master branch
    Will NOT add extra dummy commit.

  3. Squash and merge
    Will group all feature branch commits into one commit then append it in the front of the master branch
    Will add extra dummy commit.

You can find below how the master branch will look after each one of them.

enter image description here

In all cases:
We can safely DELETE the feature branch.

Solution 4

Merge squash merges a tree (a sequence of commits) into a single commit. That is, it squashes all changes made in n commits into a single commit.

Rebasing is re-basing, that is, choosing a new base (parent commit) for a tree. Maybe the mercurial term for this is more clear: they call it transplant because it's just that: picking a new ground (parent commit, root) for a tree.

When doing an interactive rebase, you're given the option to either squash, pick, edit or skip the commits you are going to rebase.

Hope that was clear!

Share:
221,344
GiH
Author by

GiH

Updated on July 08, 2022

Comments

  • GiH
    GiH almost 2 years

    I'm new to git and I'm trying to understand the difference between a squash and a rebase. As I understand it you perform a squash when doing a rebase.

  • Wayne Conrad
    Wayne Conrad about 14 years
    G is c--d--e--f--g squashed together?
  • VonC
    VonC about 14 years
    @Wayne: yes, G in those examples represent the tmp commits squashed together.
  • Alexander Bird
    Alexander Bird almost 13 years
    Isn't the data in G exactly the same as in g? So would that mean that G is the same as g except that it has a different parent?
  • VonC
    VonC almost 13 years
    @Th4wn: Since Git reasons with snapshots of a all project, G won't represent the same content than g, because of changes introduced by X.
  • VonC
    VonC almost 13 years
    Note to self: a "cleaner history" is important, considering a good Git workflow: sandofsky.com/blog/git-workflow.html
  • naught101
    naught101 over 11 years
    @VonC: not sure about that last comment. If you have a git merge --no-ff temp instead of git merge --squash temp, then you get a messier history, but you can also do things like git revert e, much more easily. It's a messy, but honest and pragmatic history, and the main branch still remains fairly clean.
  • VonC
    VonC over 11 years
    @naught101 I agree. As explained in stackoverflow.com/a/7425751/6309 though, it is also about not breaking git bisect or git blame when used too often (as in git pull --no-ff: stackoverflow.com/questions/12798767/…). There isn't one approach anyway, which is why this article described three (stackoverflow.com/questions/9107861/…)
  • Mr_and_Mrs_D
    Mr_and_Mrs_D almost 11 years
    Hmm - if say commit c involved deleting a file that was added in a so also present in X then git merge --squash would have this file in G - see here. So G and g would be different versions of the working tree ! I don't know of other pitfalls but probably git merge --squash is not the way to mirror the state of tmp on stable. See also this
  • Martin Thoma
    Martin Thoma almost 8 years
    When should I rebase and when should I squash?
  • Martin Thoma
    Martin Thoma almost 8 years
    When should I rebase and when should I squash?
  • VonC
    VonC almost 8 years
    @MartinThoma not that a rebase can also allow a squash (blog.ona.io/general/2016/02/02/…) More generally, you squash when you do not need the intermediate commits. This is a current practice when accenting a PR (Pull Request): see github.com/blog/2141-squash-your-commits
  • Yusuf
    Yusuf almost 4 years
    can you explain what is dummy commit in 2nd picture ?? I am a beginner in git.
  • ahmednabil88
    ahmednabil88 almost 4 years
    @Yusuf, it's just an extra commit which contains both branches updates, it's default commit message = "Megre branch XYZ into master"
  • Mauricio Scheffer
    Mauricio Scheffer almost 4 years
  • leticia
    leticia almost 4 years
    For "Squash and merge": there is a commit with all grouped commits plus an "extra dummy commit"?
  • ahmednabil88
    ahmednabil88 almost 4 years
    @leticia the commit with all grouped commits = the "extra dummy commit" itself, as the above graph
  • max
    max over 3 years
    It does not matter which you use but I recommend rebase. Rebase changes the parent node of the feature branch but merge does not and I recommend it because it keeps the commit structure simpler but as a git user, it makes not different. stackoverflow.com/questions/2427238/….
  • BozanicJosip
    BozanicJosip about 3 years
    Then I would argue that 'squash and merge' will not add an extra dummy commit, but rather will just 'rebase'/append the commit in front of the master branch. Dummy commit in the context you are describing it above is not the same in 1. and 3. as dummy commit in 1 is 'Merge branch XYZ into master which is producing this commit' and dummy commit in 3 is 'Squashed commits into this one commit which is not additional commit produced by merge'
  • IMSoP
    IMSoP almost 3 years
    That first diagram looks completely wrong to me. Somehow, commit D has ended up with no parent.
  • XardasLord
    XardasLord over 2 years
    Perfect diagrams and explanation!
  • rahulaga-msft
    rahulaga-msft over 2 years
    @ahmednabil88 nice explanation !! in the case of squash merge (final 3rd figure above) can we say all commits from the feature branch combined into a single commit and that single commit rebased on top of master branch like any other usual rebase merge?
  • becko
    becko about 2 years
    I don't see the difference between "Merge commit" and "Squash and merge". It seems the master branch looks identical in the final state in both cases.