How can I move around the bash commandline efficiently?

So I use the up in my shell (Bash on OSX or Ubuntu, mostly) but some of the time I know that I want the stuff after the cursor's current location. Is there any way to have the line I'm on complete with the line above? Hitting up and then left is annoying.

Any other tricks would be cool too, except for tab, which we all now about :)

Here's an example (per ~quack's request):

I type mkdir /where/the/hell/is/that/thing and then I want to cd into the same directory. But now I know about meta-b :)


Solution 1:

Try this:

mkdir /where/the/hell/is/that/thing

then press alt+.

depending on your OS and terminal you might have to type esc then .

You can also press alt+1+. to pick a specific prior argument.

This is much better than !! or !$ because you can actually see what you are about to run, and it takes less keystrokes anyway.

edit: strictly speaking it is meta not alt, so it might also work with the "windows" key depending on how your keyboard is setup..

Solution 2:

You should read through man bash on your system, particularly the section on Readline, as this is bash's interactive input mechanism. The Bash manual at gnu.org has a nice section on Commandline Editing that will help fill in some of the gaps.

By default, you can use these to move around on the commandline (just listing a few here; see the link for a full list):

  • Ctrl+a moves to the beginning of the line (or Home depending on terminal settings)
  • Ctrl+e moves to the end of the line (or End ...)
  • Meta+f moves forward a "word"
  • Meta+b moves backward a "word"

You can use these to "kill" (aka "cut") text from a line:

  • Ctrl+k kills ("cuts") text from the current cursor position to the end of the line
  • Ctrl+y yanks ("pastes") the most recently killed text back into the buffer at the cursor

So you can combine these to select a chunk of some commandline you want to repeat, kill it, then paste it to the end of your next command.


Now, to make it even more fun, let's consider bash's History Expansion. This is that !! that Studer's answer mentions. History expansion breaks down into event designators, word designators, and modifiers.

Event designators look like this (again, see the links for the full list):

  • ! - starts a history substitution
  • !n - the n-th command in bash's history list, for some integer n (works for negatives too)
  • !! - the preceeding command; equivalent to !-1
  • !string - the most recent command starting with string

Word designators select certain parts from an event. Use : to separate the event from the word designator. Words are numbered from 0 starting at the beginning of the line, and inserted into the current line separated by single spaces.

  • $ - designates the last argument (eg !!:$ is the last arg of last command; can be shortened to !$)
  • n - designates the n-th word (eg !str:2 is the 2nd arg of the most recent command starting with str; !!:0 is the command of the last command)

So, to follow up on your example, if your last command is mkdir /some/really/long/path, just running !! will rerun that command again. But you want to cd into that path instead:

$ cd !$

Now let's say you perform some other commands and then want to refer back to that path again. If that was the last mkdir command you ran, you can use this to repeat that path:

$ tar czf ~/foo.tgz !mkdir:$

Solution 3:

If you're a vi user you can put the shell into vi mode by typing set -o vi.

You can then do all kinds of powerful things on the command line after typing Esc + k.

The one I use the most is Esc+K then / any text typed after the slash will be used to search through your command line history.

e.g. Esc + k /smbclient will give the last smbclient command you typed. You can scroll through all the searches using standard vi keys (j and k for up and down but I think up and down arrows will work too).

Further help here, there are also some inbuilt bash things at that link which might be more what you're looking for.

Cheers, Stu.

Solution 4:

You can use !! which represents the last command :

% test
% echo !!
> test