Removing files with a certain extension except one file from terminal
Solution 1:
Here's the simple solution you should probably use:
mv filename.gif filename.gif.keep
rm *.gif
mv filename.gif.keep filename.gif
There's nothing special about the .keep
extension, this just makes it so that the filename temporarily doesn't end in .gif
.
If you must not rename the file (and there are scripting situations where this is important):
for X in *.gif; do
if [ "$X" != "filename.gif" ]; then
rm "$X"
fi
done
Or you can write it shorter like this:
for X in *.gif; do [ "$X" != "filename.gif" ] && rm "$X"; done
You may prefer to use find
instead; it's very powerful, you might consider it more readable, and it better handles weird filenames with characters like *
in them.
find . -maxdepth 1 -not -name 'filename.gif' -name '*.gif' -delete
I've used the -not
operator for readability, but if POSIX compliance is important--if you're not using GNU find, or if this is for a script you intend to redistribute to others or run on a variety of systems--you should use the !
operator instead:
find . -maxdepth 1 ! -name 'filename.gif' -name '*.gif' -delete
One handy thing about find
is that you can easily modify the command for case-insensitivity, so that it finds and deletes files with extensions like .GIF
too:
find . -maxdepth 1 -not -name 'filename.gif' -iname '*.gif' -delete
Please note that I've used -iname
in place of -name
for the search pattern *.gif
but I have not used it for filename.gif
. Presumably you know exactly what your file is called, and -iname
will match alternate capitalization not just in the extension, but anywhere in the filename.
All these solutions only delete files residing immediately in the current directory. They don't delete files not contained in the current directory, and they don't delete files that reside in subdirectories of the current directory.
If you want to delete files everywhere contained within the current directory (that is, including in subdirectories, and in subdirectories of those subdirectories, and so forth--files contained within the current directory or any of its descendants), use find
without maxdepth -1
:
find . -not -name 'filename.gif' -name '*.gif' -delete
Be careful with this!
You can also set other values with -maxdepth
. For example, to delete files in the current directory and its children and grandchildren but not any deeper:
find . -maxdepth 3 -not -name 'filename.gif' -name '*.gif' -delete
Just make sure you never put -delete
first, before the other expressions! You'll see I've always put -delete
at the end. If you put it at the beginning, it would be evaluated first, and all the files under .
(including files not ending in .gif
and files in deep sub-sub-sub...directories of .
) would be deleted!
For more information, see the manual pages for bash
and sh
and for the commands used in these examples: mv
, rm
, [
, and (especially) find
.
Solution 2:
If you're using bash
(the default shell), the extglob
shell option allows you to use an extended pattern matching syntax. To enable it, use the shopt
builtin command:
shopt -s extglob
(I include that line in my .bashrc
file.)
Among other things, it grants access to the !()
operator, which matches any pattern not inside the parens. For your purpose:
rm !(filename).gif
More information is available in man bash
under "Pattern Matching".
Solution 3:
Open a Terminal with Ctrl + Alt + T and type:
find . -type f -name "*.gif" -and -not -name "filename.gif" -exec rm -vf {} \;
The difference between -delete
and -exec rm -vf {} \;
is that with the second option, you will be able to see which files have been removed, because of the -v
flag. That's something that can't be done with the -delete
option. (-name -delete
will print the filenames, but rm
with -v
reveals if each file was successfully deleted.)