10 Amazing and Mysterious Uses of (!) Symbol or Operator in Linux Commands
by Avishek Kumar | Published: May 15, 2015 | Last Updated: May 18, 2015
Download Your Free eBooks NOW - 10 Free Linux eBooks for Administrators | 4 Free Shell Scripting eBooks
The
'!'
symbol or operator in Linux can be used as Logical Negation operator as well as to fetch commands from history with tweaks or to run previously run command with modification. All the commands below have been checked explicitly in bash Shell. Though I have not checked but a major of these won’t run in other shell. Here we go into the amazing and mysterious uses of '!'
symbol or operator in Linux commands.1. Run a command from history by command number.
You might not be aware of the fact that you can run a command from your history command (already/earlier executed commands). To get started first find the command number by running ‘history‘ command.
$ history
Now run a command from history just by the number at which it appears, in the output of history. Say run a command that appears at number 1551 in the output of ‘history‘ command.
$ !1551
And, it runs the command (top command in the above case), that was listed at number 1551. This way to retrieving already executed command is very helpful specially in case of those commands which are long. You just need to call it using ![Number at which it appears in the output of history command].
2. Run previously executed command as 2nd last command, 7th last command,etc.
You may run those commands which you have run previously by their running sequence being the last run command will be represented as -1, second last as -2, seventh last as -7,….
First run history command to get a list of last executed command. It is necessary to run historycommand, so that you can be sure that there is no command like
rm command > file
and others just to make sure you do not run any dangerous command accidentally. And then check Sixth last command, Eight last command and Tenth last command.$ history $ !-6 $ !-8 $ !-10
3. Pass arguments of last command that we run to the new command without retyping
I need to list the content of directory ‘/home/$USER/Binary/firefox‘ so I fired.
$ ls /home/$USER/Binary/firefox
Then I realized that I should have fired ‘ls -l‘ to see which file is executable there? So should I type the whole command again! No I don’t need. I just need to carry the last argument to this new command as:
$ ls -l !$
Here
!$
will carry arguments passed in last command to this new command.4. How to handle two or more arguments using (!)
Let’s say I created a text file 1.txt on the Desktop.
$ touch /home/avi/Desktop/1.txt
and then copy it to ‘/home/avi/Downloads‘ using complete path on either side with cp command.
$ cp /home/avi/Desktop/1.txt /home/avi/downloads
Now we have passed two arguments with cp command. First is ‘/home/avi/Desktop/1.txt‘ and second is ‘/home/avi/Downloads‘, lets handle them differently, just execute
echo [arguments]
to print both arguments differently.$ echo “1st Argument is : !^” $ echo “2nd Argument is : !cp:2”
Note 1st argument can be printed as
“!^”
and rest of the arguments can be printed by executing“![Name_of_Command]:[Number_of_argument]”
.
In the above example the first command was ‘cp‘ and 2nd argument was needed to print. Hence
“!cp:2”
, if any command say xyz is run with 5 arguments and you need to get 4th argument, you may use “!xyz:4”
, and use it as you like. All the arguments can be accessed by “!*”
.5. Execute last command on the basis of keywords
We can execute the last executed command on the basis of keywords. We can understand it as follows:
$ ls /home > /dev/null [Command 1] $ ls -l /home/avi/Desktop > /dev/null [Command 2] $ ls -la /home/avi/Downloads > /dev/null [Command 3] $ ls -lA /usr/bin > /dev/null [Command 4]
Here we have used same command (ls) but with different switches and for different folders. Moreover we have sent to output of each command to ‘/dev/null‘ as we are not going to deal with the output of the command also the console remains clean.
Now Execute last run command on the basis of keywords.
$ ! ls [Command 1] $ ! ls -l [Command 2] $ ! ls -la [Command 3] $ ! ls -lA [Command 4]
Check the output and you will be astonished that you are running already executed commands just by
ls
keywords.6. The power of !! Operator
You can run/alter your last run command using
(!!)
. It will call the last run command with alter/tweak in the current command. Lets show you the scenario
Last day I run a one-liner script to get my private IP so I run,
$ ip addr show | grep inet | grep -v 'inet6'| grep -v '127.0.0.1' | awk '{print $2}' | cut -f1 -d/
Then suddenly I figured out that I need to redirect the output of the above script to a file ip.txt, so what should I do? Should I retype the whole command again and redirect the output to a file? Well an easy solution is to use
UP
navigation key and add '> ip.txt'
to redirect the output to a file as.$ ip addr show | grep inet | grep -v 'inet6'| grep -v '127.0.0.1' | awk '{print $2}' | cut -f1 -d/ > ip.txt
Thanks to the life Savior
UP
navigation key here. Now consider the below condition, the next time I run below one-liner script.$ ifconfig | grep "inet addr:" | awk '{print $2}' | grep -v '127.0.0.1' | cut -f2 -d:
As soon as I run script, the bash prompt returned an error with the message
“bash: ifconfig: command not found”
, It was not difficult for me to guess I run this command as user where it should be run as root.
So what’s the solution? It is difficult to login to root and then type the whole command again! Also (UP Navigation Key) in last example didn’t came to rescue here. So? We need to call
“!!”
without quotes, which will call the last command for that user.$ su -c “!!” root
Here su is switch user which is root,
-c
is to run the specific command as the user and the most important part !!
will be replaced by command and last run command will be substituted here. Yeah! You need to provide root password.
I make use of
!!
mostly in following scenarios,
1. When I run apt-get command as normal user, I usually get an error saying you don’t have permission to execute.
$ apt-get upgrade && apt-get dist-upgrade
Opps error…don’t worry execute below command to get it successful..
$ su -c !!
Same way I do for,
$ service apache2 start or $ /etc/init.d/apache2 start or $ systemctl start apache2
OOPS User not authorized to carry such task, so I run..
$ su -c 'service apache2 start' or $ su -c '/etc/init.d/apache2 start' or $ su -c 'systemctl start apache2'
7. Run a command that affects all the file except ![FILE_NAME]
The
!
(Logical NOT) can be used to run the command on all the files/extension except that is behind'!'
.
A. Remove all the files from a directory except the one the name of which is 2.txt.
$ rm !(2.txt)
B. Remove all the file type from the folder except the one the extension of which is ‘pdf‘.
$ $ rm !(*.pdf)
8. Check if a directory (say /home/avi/Tecmint)exist or not? Printf if the said directory exist or not.
Here we will use
'! -d'
to validate if the directory exist or not followed by Logical AND Operator(&&)
to print that directory does not exist and Logical OR Operator (||)
to print the directory is present.
Logic is, when the output of
[ ! -d /home/avi/Tecmint ]
is 0, it will execute what lies beyond Logical AND else it will go to Logical OR (||)
and execute what lies beyond Logical OR.$ [ ! -d /home/avi/Tecmint ] && printf '\nno such /home/avi/Tecmint directory exist\n' || printf '\n/home/avi/Tecmint directory exist\n'
9. Check if a directory exist or not? If not exit the command.
Similar to the above condition, but here if the desired directory doesn’t exist it will exit the command.
$ [ ! -d /home/avi/Tecmint ] && exit
10. Create a directory (say test) in your home directory if it does not exist.
A general implementation in Scripting Language where if the desired directory does not exist, it will create one.
[ ! -d /home/avi/Tecmint ] && mkdir /home/avi/Tecmint
That’s all for now. If you know or come across any other use of
'!'
which is worth knowing, you may like to provide us with your suggestion in the feedback. Keep connected!
If you have any questions or problems regarding this article and want help within 24 Hours? Ask Now
Support TecMint: Did you find this tutorial helpful?. Please help to keep it alive by donating. Every cent counts! - Donate Now
Tags: Linux Tricks
Avishek Kumar
I am a major in computer science, love to research nix. I love to write codes and scripts, review distros, experiment Foss Technologies, write technical articles, Hack, of course Ethically. I am working as System Administrator (nix) for a NGO.
Your name can also be listed here. Got a tip? Submit it here to become an TecMint author.
Receive Your Free Complimentary eBook NOW! - Linux Kernel In A Nutshell - Configuration and Building
- NEXT STORYHow to Configure Network (NIC) Bonding/Teaming on Debian Linux
- PREVIOUS STORYInstall ‘Tails 1.4’ Linux Operating System to Preserve Privacy and Anonymity
YOU MAY ALSO LIKE...
12 RESPONSES
LEAVE A REPLY
DOWNLOAD FREE LINUX EBOOKS
- Complete Linux Command Line Cheat Sheet
- The GNU/Linux Advanced Administration Guide
- Securing & Optimizing Linux Servers
- Linux Patch Management: Keeping Linux Up To Date
- Introduction to Linux – A Hands on Guide
- Understanding the Linux® Virtual Memory Manager
- Linux Bible – Packed with Updates and Exercises
- A Newbie’s Getting Started Guide to Linux
- Linux from Scratch – Create Your Own Linux OS
- Linux Shell Scripting Cookbook, Second Edition
- Securing & Optimizing Linux: The Hacking Solution
- User Mode Linux – Understanding and Administration
1013 grep tornado /usr/share/dict/words
1014 grep hurricane /usr/share/dict/words
1015 wc -l /usr/share/dict/words
Your Suggestion is taken into account.
Thanks for your feedback.
Thanks for the wonderful piece of Information. Keep connected and keep us aware of such context.
^ is at the begging expression
$ is at the end expression
:number any number parameter
touch a.txt b.txt c.txt
echo !^ –> display first parameter
echo !:1 –> also display first parameter
echo !:2 –> display second parameter
echo !:3 –> display third parameter
echo !$ –> display last (in our case 3th) parameter
echo !* –> display all parameters
$ echo “2nd Argument is : !cp:2”
When you invoke “! ls…”, it always picks the last ls command you executed, just appends your switches at the end (after /dev/null).
At least that’s still representative of what it actually does, unlike when most people refer to it as “Super User”, thinking it only grants root user status, which is utterly incorrect.
thanks corrected in the write up..