Git reset (soft, mixed, hard)

What are Git reset modes (soft, mixed, hard)?

Git reset (soft, mixed, hard) are different modes of the reset command, each affecting the working directory, staging area, and commit history differently. Soft keeps changes staged, mixed unstages changes, and hard discards all changes, providing flexibility in undoing commits.

Git is a distributed version control system that allows multiple people to work on a project at the same time without overwriting each other's changes. It's a crucial tool for modern software development, and understanding its various commands and features is vital for any software engineer. One such command is 'git reset', which can be used in three different modes: soft, mixed, and hard. This article will delve into the intricacies of these three modes, providing a thorough understanding of their functionalities and use cases.

Before we dive into the details, it's important to have a basic understanding of how Git works. Git tracks changes to a project over time in a series of snapshots, or commits. Each commit represents a specific state of the project, and Git allows you to move back and forth between these states as needed. The 'git reset' command is one way to navigate these states, but its effects can be quite different depending on whether it's used in soft, mixed, or hard mode.

Definition

The 'git reset' command is used to move the HEAD pointer to a specific commit. The HEAD pointer is a reference to the current commit, and moving it effectively changes the state of the project. However, the way this state change is handled depends on whether the reset is soft, mixed, or hard.

In soft mode, 'git reset' moves the HEAD pointer but leaves the staging area and the working directory unchanged. This means that any changes that were staged for the next commit will remain staged, and any changes in the working directory that haven't been staged will also remain. This is useful when you want to keep your changes but move the HEAD pointer to a different commit.

Soft Reset

A soft reset in Git is the least destructive type of reset. It moves the HEAD pointer to a specific commit, but it leaves the staging area and the working directory unchanged. This means that any changes that were staged for the next commit will remain staged, and any changes in the working directory that haven't been staged will also remain. This is useful when you want to keep your changes but move the HEAD pointer to a different commit.

For example, if you've made several commits but realize that they should all be one single commit, you can use a soft reset to move the HEAD pointer back to the commit before the first one in the series. Then, you can commit all the changes as one big commit. The command for a soft reset is 'git reset --soft [commit]'.

Mixed Reset

A mixed reset, which is the default mode of 'git reset', moves the HEAD pointer and also resets the staging area to match the specified commit, but it leaves the working directory unchanged. This means that any changes in the working directory that haven't been staged will remain, but any changes that were staged for the next commit will be unstaged. This is useful when you want to unstage some changes but keep them in your working directory.

For example, if you've staged several changes for the next commit but then decide that some of them should be in a separate commit, you can use a mixed reset to unstage the changes. Then, you can stage and commit the changes separately. The command for a mixed reset is 'git reset --mixed [commit]', or simply 'git reset [commit]'.

Hard Reset

A hard reset in Git is the most destructive type of reset. It moves the HEAD pointer, resets the staging area, and also resets the working directory to match the specified commit. This means that any changes in the staging area or the working directory will be lost. This is useful when you want to discard all your changes and go back to a specific commit.

For example, if you've made several changes but then decide that they're all wrong and you want to go back to the way things were at a previous commit, you can use a hard reset. Be careful, though, because this will permanently discard all the changes you've made since that commit. The command for a hard reset is 'git reset --hard [commit]'.

Explanation

To fully understand the 'git reset' command, it's helpful to have a clear picture of Git's three main areas: the Git directory, the staging area, and the working directory. The Git directory is where Git stores the metadata and object database for your project. This is the most important part of Git, and it's what is copied when you clone a repository from another computer.

The staging area, also known as the index, is a file that stores information about what will go into your next commit. You can think of the staging area as a prep table where Git will format and sum up the changes for the next commit. The working directory is a single checkout of one version of the project. These files are pulled out of the compressed database in the Git directory and placed on disk for you to use or modify.

Git Directory

The Git directory is the most important part of Git. It's where Git stores the metadata and object database for your project. This includes all the information about the commits, the references to the commits, and the objects that represent the content of the files, directories, and the commits themselves. When you clone a repository from another computer, it's the Git directory that gets copied.

When you run 'git reset', it first moves the HEAD pointer in the Git directory to the specified commit. This changes the current commit, which is the commit that the next commit will be based on. However, the effects of this move depend on whether the reset is soft, mixed, or hard.

Staging Area

The staging area, also known as the index, is a file that stores information about what will go into your next commit. When you run 'git add', it adds changes to the staging area, marking them for inclusion in the next commit. Conversely, when you run 'git reset', it can remove changes from the staging area, marking them for exclusion from the next commit.

In a soft reset, the staging area is left unchanged, so any changes that were staged will remain staged. In a mixed reset, the staging area is reset to match the specified commit, so any changes that were staged will be unstaged. In a hard reset, the staging area is also reset to match the specified commit, but in addition, the working directory is reset as well, discarding any unstaged changes.

Working Directory

The working directory is a single checkout of one version of the project. These files are pulled out of the compressed database in the Git directory and placed on disk for you to use or modify. When you make changes to these files, those changes are considered "unstaged" until you run 'git add' to stage them.

In a soft or mixed reset, the working directory is left unchanged, so any unstaged changes will remain. However, in a hard reset, the working directory is reset to match the specified commit, discarding any unstaged changes. This is why a hard reset is considered destructive: it permanently discards changes, whether they're staged or not.

History

Git was created by Linus Torvalds in 2005 for development of the Linux kernel, with other kernel developers contributing to its initial development. It's named after the British English slang word for "unpleasant person", which Torvalds said he chose because he's "an egotistical bastard, and I name all my projects after myself. First 'Linux', now 'Git'".

The 'git reset' command has been a part of Git since its early days, and its functionality has remained largely unchanged. However, the understanding and usage of this command have evolved over time as more and more people have started using Git and sharing their knowledge and best practices.

Use Cases

The 'git reset' command is incredibly versatile and can be used in a variety of situations. Here are a few common use cases:

Unstaging Changes

If you've staged some changes with 'git add' but then decide that you don't want to commit them yet, you can use 'git reset' to unstage them. This is typically done with a mixed reset, which resets the staging area but leaves the working directory unchanged. The command is 'git reset [file]', which will unstage the specified file, or 'git reset' with no file specified, which will unstage all changes.

For example, if you've staged a file called 'file1.txt' but then decide that you don't want to commit it yet, you can run 'git reset file1.txt' to unstage it. The changes to 'file1.txt' will still be in your working directory, but they won't be included in the next commit until you stage them again with 'git add'.

Discarding Changes

If you've made some changes that you want to discard, you can use 'git reset' to do so. This is typically done with a hard reset, which resets both the staging area and the working directory. The command is 'git reset --hard [commit]', which will discard all changes since the specified commit.

For example, if you've made a bunch of changes but then decide that they're all wrong and you want to go back to the way things were at the last commit, you can run 'git reset --hard HEAD' to discard all changes. Be careful, though, because this will permanently discard all changes, whether they're staged or not.

Combining Commits

If you've made several commits but then decide that they should all be one single commit, you can use 'git reset' to combine them. This is typically done with a soft reset, which moves the HEAD pointer but leaves the staging area and the working directory unchanged. The command is 'git reset --soft [commit]', which will move the HEAD pointer to the specified commit and leave all changes since that commit staged for the next commit.

For example, if you've made three commits but then decide that they should all be one commit, you can run 'git reset --soft HEAD~3' to move the HEAD pointer back three commits. Then, all changes from those three commits will be staged for the next commit, and you can run 'git commit' to commit them as one big commit.

Specific Examples

Let's look at some specific examples to better understand how 'git reset' works in different modes.

Soft Reset Example

Suppose you have a Git repository with the following history:


A -- B -- C (HEAD)

You've made three commits (A, B, and C), and the HEAD pointer is currently at commit C. You've also made some changes and staged them for the next commit.

Now, you decide that the changes you've staged should actually be part of commit B, not a new commit. To do this, you can run 'git reset --soft B', which will move the HEAD pointer to commit B and leave the changes staged:


A -- B (HEAD)
\
C

Now, you can run 'git commit' to commit the staged changes as part of commit B.

Mixed Reset Example

Suppose you have a Git repository with the following history:


A -- B -- C (HEAD)

You've made three commits (A, B, and C), and the HEAD pointer is currently at commit C. You've also made some changes and staged them for the next commit.

Now, you decide that the changes you've staged should be unstaged. To do this, you can run 'git reset --mixed HEAD', which will move the HEAD pointer to commit C (i.e., it stays where it is) and unstage the changes:


A -- B -- C (HEAD)

Now, the changes are in your working directory but not staged for the next commit.

Hard Reset Example

Suppose you have a Git repository with the following history:


A -- B -- C (HEAD)

You've made three commits (A, B, and C), and the HEAD pointer is currently at commit C. You've also made some changes and staged them for the next commit.

Now, you decide that the changes you've made and staged are all wrong and you want to discard them. To do this, you can run 'git reset --hard HEAD', which will move the HEAD pointer to commit C (i.e., it stays where it is), unstage the changes, and discard them:


A -- B -- C (HEAD)

Now, the changes are gone, and your project is back to the state it was in at commit C.

Conclusion

The 'git reset' command is a powerful tool in Git, allowing you to move the HEAD pointer, reset the staging area, and reset the working directory in various combinations. Whether you want to unstage some changes, discard all changes, or combine several commits into one, 'git reset' has you covered.

However, with great power comes great responsibility. Be careful when using 'git reset', especially in hard mode, as it can permanently discard changes. Always make sure you understand what a command will do before you run it. And remember, Git is a tool to help you, not to hinder you. If a command seems too complicated or dangerous, there's probably a safer and simpler way to accomplish what you want.

Join other high-impact Eng teams using Graph
Ready to join the revolution?
Join other high-impact Eng teams using Graph
Ready to join the revolution?

Build more, chase less

Add to Slack