git rm
will remove entries from the staging area. This is a bit different fromgit reset HEAD
which "unstages" files. By "unstage" I mean it reverts the staging area to what was there before we started modifying things.git rm
on the other hand just kicks the file off the stage entirely, so that it's not included in the next commit snapshot, thereby effectively deleting it.By default, a
git rm file
will remove the file from the staging area entirely and also off your disk > (the working directory). To leave the file in the working directory, you can usegit rm --cached
.
But what exactly is the difference between git rm --cached asd
and git reset head -- asd
?
There are three places where a file, say, can be - the tree, the index and the working copy. When you just add a file to a folder, you are adding it to the working copy.
When you do something like
git add file
you add it to the index. And when you commit it, you add it to the tree as well.It will probably help you to know the three more common flags in git reset:
Now, when you do something like
git reset HEAD
- what you are actually doing isgit reset HEAD --mixed
and it will "reset" the index to the state it was before you started adding files / adding modifications to the index ( viagit add
) In this case, the working copy and the index ( or staging ) were in sync, but you made the HEAD and the index to be in sync after the reset.git rm
on the other hand removes a file from the working directory and the index and when you commit, the file is removed from the tree as well.git rm --cached
however removes the file from index alone and keeps it in your working copy. This is the exact opposite ofgit add file
In this case, you made index to be different from the HEAD and the working, in it that the HEAD has the previously committed version of the file, working copy had the las modification if any or content from HEAD of the file and you removed the file from the index. A commit now will sync the index and tree and the file will be removed.git rm --cached file
will remove the file from the stage. That is, when you commit the file will be removed.git reset HEAD -- file
will simply reset file in the staging area to the state where it was on the HEAD commit, i.e. will undo any changes you did to it since last commiting. If that change happens to be newly adding the file, then they will be equivalent.Perhaps an example will help:
versus
Note that if you haven't changed anything else, the second commit won't actually do anything.