2542

In a Unix shell, if I want to combine stderr and stdout into the stdout stream for further manipulation, I can append the following on the end of my command:

2>&1

So, if I want to use head on the output from g++, I can do something like this:

g++ lots_of_errors 2>&1 | head

so I can see only the first few errors.

I always have trouble remembering this, and I constantly have to go look it up, and it is mainly because I don't fully understand the syntax of this particular trick.

Can someone break this up and explain character by character what 2>&1 means?

Ricardo
  • 2,758
  • 2
  • 23
  • 42
Tristan Havelick
  • 58,645
  • 19
  • 52
  • 64
  • 55
    @dbr I don't think it's just bash - I believe it's a bourne shell thing; hence sh, bash, ksh, ash, dash, etc. – guns May 03 '09 at 23:49
  • 8
    This is part of the redirection paragraph describing POSIX-compliant shells, or POSIX shell for short. ksh is a POSIX shell for example. See:http://pubs.opengroup.org/onlinepubs/009695399/utilities/xcu_chap02.html#tag_02_07 – jim mcnamara Apr 04 '13 at 02:55
  • 14
    This construct also works on Windows. – Vadzim Oct 22 '13 at 13:45
  • 4
    It's generally better doing `2>&1` than [2>/dev/null](http://stackoverflow.com/q/14423442/1765658) ;-) – F. Hauri Dec 08 '13 at 12:11
  • 12
    I thought I'd mention that `|&` is shorthand for `2>&1 |` if you're using zsh. I can't speak to whether that applies to other bourne-like shells or if it's a zsh only feature. – chrixian Dec 17 '13 at 05:20
  • 2
    @chrixian `|&` is also in Bash, great feature. Not POSIX it seems though (?) – Ciro Santilli新疆棉花TRUMP BAN BAD Jul 27 '15 at 10:30
  • 1
    you can check this answer https://stackoverflow.com/questions/10508843/what-is-dev-null-21/42919998#answer-42919998 – Vishrant Jul 19 '18 at 17:32

18 Answers18

2870

File descriptor 1 is the standard output (stdout).
File descriptor 2 is the standard error (stderr).

Here is one way to remember this construct (although it is not entirely accurate): at first, 2>1 may look like a good way to redirect stderr to stdout. However, it will actually be interpreted as "redirect stderr to a file named 1". & indicates that what follows and precedes is a file descriptor and not a filename. So the construct becomes: 2>&1.

Consider >& as redirect merger operator.

Smart Manoj
  • 3,837
  • 2
  • 24
  • 45
Ayman Hourieh
  • 113,954
  • 21
  • 138
  • 113
  • 1
    does this make any sense to you, `java... 2&1 >> data.log`, I saw one of my colleague did this? – Thang Pham Jul 26 '11 at 19:53
  • 2
    >> means append the output to the data.log file. – Khue Vu Nov 12 '11 at 05:53
  • 3
    @ThangPham mean STDOUT appended to data.log **and** STDERR dumped to STDOUT (console or next command) see my answer – F. Hauri Jan 17 '13 at 01:02
  • 1
    This looks accurate to me. How is it 'not entirely accurate'? – Bad Request Jul 03 '13 at 04:51
  • 315
    but then shouldn't it rather be `&2>&1`? – dokaspar Sep 04 '13 at 06:12
  • 361
    @Dominik: Nope, `&` is only interpreted to mean "file descriptor" in the context of redirections. Writing `command &2>&` is parsed as `command &` and `2>&1`, i.e. "run `command` in the background, then run the command `2` and redirect its stdout into its stdout". – Adam Rosenfield Jan 28 '14 at 00:02
  • 31
    Why did they pick such arcane stuff as this? Just curious. – CommaToast May 17 '14 at 04:00
  • 99
    But how would you redirect stderr to a file named '&1'? – Martín Fixman Nov 04 '14 at 17:07
  • 149
    @Martin: `2>'&1'` –  Mar 14 '16 at 10:46
  • 1
    any idea why this interacts with grep highlighting? `nginx -V 2>&1 | grep auth_request`, highlights the word _auth_request_ in red but `nginx -V | grep auth_request` doesn't – JonnyRaa Sep 28 '17 at 09:39
  • 5
    so & meanings file descriptor and also go to the background?? Linux needs to stop using the same symbol too confusing!! – bakalolo Jan 11 '18 at 02:11
  • 4
    In case this helps for mnemonics for someone, the way I think of this connects with the C-language `&` operator, "address of" -- so, `2>&1` as a whole I read in my head as something resembling "(file descriptor) 2 redirects to the address of (file descriptor) 1". I.e. *to where fd1 is currently going*, as of the time the shell reads that 4-character sequence. (Because it can change. For fun, try: `((echo 1/stdout; echo 2/stderr >&2) 2>&1 > /dev/tty) | grep --color .` -- it'll colorize what started out as standard error ("2/stderr").) – lindes May 05 '18 at 01:54
  • 1
    What happens if I do `2> &1` (space after `>`) instead of `2>&1`? I'm not trying it out myself over the fear of breaking stuff. – radiantshaw Aug 26 '18 at 16:13
  • @radiantshaw It just gives an error message. In bash: `foo 2>&1` outputs "-bash: foo: command not found", and `foo 2> &1` outputs "-bash: syntax error near unexpected token '&'". Shells are often picky about whitespace around symbols. @Aaron Franke Yes, `foo 2>\&1` creates a new file named "&1", containing the above error text. – Beejor May 04 '19 at 22:34
  • 2
    why did you say it's not entirely accurate? What detail did you jump over? – Fakher Mokadem Oct 13 '19 at 11:08
  • Re "*Why did they pick such arcane stuff as this?*", It's actually good to use related tokens for related things. In bash, ` – ikegami May 08 '20 at 21:33
  • 1
    @CommaToast Exactly. Everybody is telling me "Magic constants are bad in programming, avoid them, use names instead". And we have hard-coded file descriptor integers in the core of an operating system. Yack. Why couldn't they just use keywords like `stderr > stdout`? Another inconsistency - `>` usually means "replace". So, `2>&1` can be confused with replacing all output with error output. Why not use `>>` for appending, as we do for files? So, I vote for `stderr >> stdout`. – JustAMartin Sep 04 '20 at 20:26
  • 1
    @CommaToast You could say the same about just about any shell operations. Arcane symbols, overloaded operators, one-letter flags... – user3932000 Oct 10 '20 at 20:41
  • 2
    As a mnemonic, if you know C, following pointer arithmetics, you can think of it as "contents of 2 (`2`) go (`>`) to where 1 is, i.e. to _the memory address_ of 1 (`&1`)". – xmar Nov 25 '20 at 20:08
721
echo test > afile.txt

redirects stdout to afile.txt. This is the same as doing

echo test 1> afile.txt

To redirect stderr, you do:

echo test 2> afile.txt

>& is the syntax to redirect a stream to another file descriptor - 0 is stdin, 1 is stdout, and 2 is stderr.

You can redirect stdout to stderr by doing:

echo test 1>&2 # or echo test >&2

Or vice versa:

echo test 2>&1

So, in short... 2> redirects stderr to an (unspecified) file, appending &1 redirects stderr to stdout.

Peter Mortensen
  • 28,342
  • 21
  • 95
  • 123
dbr
  • 153,498
  • 65
  • 266
  • 333
  • 5
    does this make any sense to you, `java ... 2&1 >> data.log`, I saw one of my colleague did this? – Thang Pham Jul 26 '11 at 19:53
  • 6
    @Harry that looks like either a shell that isn't bash, or a typo.. `cmd 2>&1 >> somefile.log` will append stdout/stderr to a file - it's basically the same as above, with `>> file` to append – dbr Jul 27 '11 at 00:38
  • 80
    @dbr `cmd 2>&1 >>file` does not redirect stderr to the file, but `cmd >> file 2>&1` does. Order matters. In the first case, stderr is redirected to the stdout of the shell (possibly a tty if the command is entered interactively), and then stdout is directed to the file. In the second case, stdout is directed to the file, and then stderr is directed to the same place. – William Pursell Jul 19 '13 at 13:15
  • 3
    I like the answer above, but it could be a touch clearer. "2>&1" redirects stderr to the target of stdout. So if you have something like "ls -l >> directoryContents 2>&1" The result will be a file named directoryContents will have the contents of the working directory appended to it. If there are any errors in execution: the error messages will also be appended to the directoryContents file, as they occur. – Max West Apr 17 '15 at 17:36
  • 1
    Is `0(or 1,2)>&0(or 1,2)` like an option to control the output? Is `echo test >test.log 2>&1` same as `echo test 2>&1 >test.log`? – Simin Jie Jun 01 '18 at 12:58
350

Some tricks about redirection

Some syntax particularity about this may have important behaviours. There is some little samples about redirections, STDERR, STDOUT, and arguments ordering.

1 - Overwriting or appending?

Symbol > means redirection.

  • > means send to as a whole completed file, overwriting target if exist (see noclobber bash feature at #3 later).
  • >> means send in addition to would append to target if exist.

In any case, the file would be created if they not exist.

2 - The shell command line is order dependent!!

For testing this, we need a simple command which will send something on both outputs:

$ ls -ld /tmp /tnt
ls: cannot access /tnt: No such file or directory
drwxrwxrwt 118 root root 196608 Jan  7 11:49 /tmp

$ ls -ld /tmp /tnt >/dev/null
ls: cannot access /tnt: No such file or directory

$ ls -ld /tmp /tnt 2>/dev/null
drwxrwxrwt 118 root root 196608 Jan  7 11:49 /tmp

(Expecting you don't have a directory named /tnt, of course ;). Well, we have it!!

So, let's see:

$ ls -ld /tmp /tnt >/dev/null
ls: cannot access /tnt: No such file or directory

$ ls -ld /tmp /tnt >/dev/null 2>&1

$ ls -ld /tmp /tnt 2>&1 >/dev/null
ls: cannot access /tnt: No such file or directory

The last command line dumps STDERR to the console, and it seem not to be the expected behaviour... But...

If you want to make some post filtering about standard output, error output or both:

$ ls -ld /tmp /tnt | sed 's/^.*$/<-- & --->/'
ls: cannot access /tnt: No such file or directory
<-- drwxrwxrwt 118 root root 196608 Jan  7 12:02 /tmp --->

$ ls -ld /tmp /tnt 2>&1 | sed 's/^.*$/<-- & --->/'
<-- ls: cannot access /tnt: No such file or directory --->
<-- drwxrwxrwt 118 root root 196608 Jan  7 12:02 /tmp --->

$ ls -ld /tmp /tnt >/dev/null | sed 's/^.*$/<-- & --->/'
ls: cannot access /tnt: No such file or directory

$ ls -ld /tmp /tnt >/dev/null 2>&1 | sed 's/^.*$/<-- & --->/'

$ ls -ld /tmp /tnt 2>&1 >/dev/null | sed 's/^.*$/<-- & --->/'
<-- ls: cannot access /tnt: No such file or directory --->

Notice that the last command line in this paragraph is exactly same as in previous paragraph, where I wrote seem not to be the expected behaviour (so, this could even be an expected behaviour).

Well, there is a little tricks about redirections, for doing different operation on both outputs:

$ ( ls -ld /tmp /tnt | sed 's/^/O: /' >&9 ) 9>&2  2>&1  | sed 's/^/E: /'
O: drwxrwxrwt 118 root root 196608 Jan  7 12:13 /tmp
E: ls: cannot access /tnt: No such file or directory

Note: &9 descriptor would occur spontaneously because of ) 9>&2.

Addendum: nota! With the new version of (>4.0) there is a new feature and more sexy syntax for doing this kind of things:

$ ls -ld /tmp /tnt 2> >(sed 's/^/E: /') > >(sed 's/^/O: /')
O: drwxrwxrwt 17 root root 28672 Nov  5 23:00 /tmp
E: ls: cannot access /tnt: No such file or directory

And finally for such a cascading output formatting:

$ ((ls -ld /tmp /tnt |sed 's/^/O: /' >&9 ) 2>&1 |sed 's/^/E: /') 9>&1| cat -n
     1  O: drwxrwxrwt 118 root root 196608 Jan  7 12:29 /tmp
     2  E: ls: cannot access /tnt: No such file or directory

Addendum: nota! Same new syntax, in both ways:

$ cat -n <(ls -ld /tmp /tnt 2> >(sed 's/^/E: /') > >(sed 's/^/O: /'))
     1  O: drwxrwxrwt 17 root root 28672 Nov  5 23:00 /tmp
     2  E: ls: cannot access /tnt: No such file or directory

Where STDOUT go through a specific filter, STDERR to another and finally both outputs merged go through a third command filter.

3 - A word about noclobber option and >| syntax

That's about overwriting:

While set -o noclobber instruct bash to not overwrite any existing file, the >| syntax let you pass through this limitation:

$ testfile=$(mktemp /tmp/testNoClobberDate-XXXXXX)

$ date > $testfile ; cat $testfile
Mon Jan  7 13:18:15 CET 2013

$ date > $testfile ; cat $testfile
Mon Jan  7 13:18:19 CET 2013

$ date > $testfile ; cat $testfile
Mon Jan  7 13:18:21 CET 2013

The file is overwritten each time, well now:

$ set -o noclobber

$ date > $testfile ; cat $testfile
bash: /tmp/testNoClobberDate-WW1xi9: cannot overwrite existing file
Mon Jan  7 13:18:21 CET 2013

$ date > $testfile ; cat $testfile
bash: /tmp/testNoClobberDate-WW1xi9: cannot overwrite existing file
Mon Jan  7 13:18:21 CET 2013

Pass through with >|:

$ date >| $testfile ; cat $testfile
Mon Jan  7 13:18:58 CET 2013

$ date >| $testfile ; cat $testfile
Mon Jan  7 13:19:01 CET 2013

Unsetting this option and/or inquiring if already set.

$ set -o | grep noclobber
noclobber           on

$ set +o noclobber

$ set -o | grep noclobber
noclobber           off

$ date > $testfile ; cat $testfile
Mon Jan  7 13:24:27 CET 2013

$ rm $testfile

4 - Last trick and more...

For redirecting both output from a given command, we see that a right syntax could be:

$ ls -ld /tmp /tnt >/dev/null 2>&1

for this special case, there is a shortcut syntax: &> ... or >&

$ ls -ld /tmp /tnt &>/dev/null

$ ls -ld /tmp /tnt >&/dev/null

Nota: if 2>&1 exist, 1>&2 is a correct syntax too:

$ ls -ld /tmp /tnt 2>/dev/null 1>&2

4b- Now, I will let you think about:

$ ls -ld /tmp /tnt 2>&1 1>&2  | sed -e s/^/++/
++/bin/ls: cannot access /tnt: No such file or directory
++drwxrwxrwt 193 root root 196608 Feb  9 11:08 /tmp/

$ ls -ld /tmp /tnt 1>&2 2>&1  | sed -e s/^/++/
/bin/ls: cannot access /tnt: No such file or directory
drwxrwxrwt 193 root root 196608 Feb  9 11:08 /tmp/

4c- If you're interested in more information

You could read the fine manual by hitting:

man -Len -Pless\ +/^REDIRECTION bash

in a console ;-)

KyleMit
  • 45,382
  • 53
  • 367
  • 544
F. Hauri
  • 51,421
  • 13
  • 88
  • 109
  • 5
    *Further reading:* If you liked this, you may apreciate: [How redirection abuse could give strange behaviours](http://stackoverflow.com/q/14423442/1765658) – F. Hauri Dec 10 '13 at 21:14
  • 1
    *Further reading ||*: A [function to store both output into separated variables](http://stackoverflow.com/a/13807118/1765658) – F. Hauri Jan 18 '15 at 12:37
146

I found this brilliant post on redirection: All about redirections

Redirect both standard output and standard error to a file

$ command &>file

This one-liner uses the &> operator to redirect both output streams - stdout and stderr - from command to file. This is Bash's shortcut for quickly redirecting both streams to the same destination.

Here is how the file descriptor table looks like after Bash has redirected both streams:

Enter image description here

As you can see, both stdout and stderr now point to file. So anything written to stdout and stderr gets written to file.

There are several ways to redirect both streams to the same destination. You can redirect each stream one after another:

$ command >file 2>&1

This is a much more common way to redirect both streams to a file. First stdout is redirected to file, and then stderr is duplicated to be the same as stdout. So both streams end up pointing to file.

When Bash sees several redirections it processes them from left to right. Let's go through the steps and see how that happens. Before running any commands, Bash's file descriptor table looks like this:

Enter image description here

Now Bash processes the first redirection >file. We've seen this before and it makes stdout point to file:

Enter image description here

Next Bash sees the second redirection 2>&1. We haven't seen this redirection before. This one duplicates file descriptor 2 to be a copy of file descriptor 1 and we get:

Enter image description here

Both streams have been redirected to file.

However be careful here! Writing

command >file 2>&1

is not the same as writing:

$ command 2>&1 >file

The order of redirects matters in Bash! This command redirects only the standard output to the file. The stderr will still print to the terminal. To understand why that happens, let's go through the steps again. So before running the command, the file descriptor table looks like this:

Enter image description here

Now Bash processes redirections left to right. It first sees 2>&1 so it duplicates stderr to stdout. The file descriptor table becomes:

Enter image description here

Now Bash sees the second redirect, >file, and it redirects stdout to file:

Enter image description here

Do you see what happens here? Stdout now points to file, but the stderr still points to the terminal! Everything that gets written to stderr still gets printed out to the screen! So be very, very careful with the order of redirects!

Also note that in Bash, writing

$ command &>file

is exactly the same as:

$ command >&file

Peter Mortensen
  • 28,342
  • 21
  • 95
  • 123
Deen John
  • 2,598
  • 3
  • 23
  • 27
  • 4
    The last two are different if "command" ends in a number, as then that is taken as optional file descriptor for `>&` – M.M Jan 23 '17 at 12:09
  • Very nice drawing and explanation! Could you elaborate what "duplicate" really means? You mentioned, "This one [2>&1] duplicates file descriptor 2 to be a copy of file descriptor 1". It sounds like stderr gets duplicated to stdout. But if it is the case, should I also see err though `/dev/tty0`? – HCSF Jul 22 '19 at 02:35
  • 1
    This is a very nice explanation with visual. If I became the one who ask this question, I will mark this as accepted answer. – MaXi32 Oct 18 '20 at 03:43
101

The numbers refer to the file descriptors (fd).

  • Zero is stdin
  • One is stdout
  • Two is stderr

2>&1 redirects fd 2 to 1.

This works for any number of file descriptors if the program uses them.

You can look at /usr/include/unistd.h if you forget them:

/* Standard file descriptors.  */
#define STDIN_FILENO    0   /* Standard input.  */
#define STDOUT_FILENO   1   /* Standard output.  */
#define STDERR_FILENO   2   /* Standard error output.  */

That said I have written C tools that use non-standard file descriptors for custom logging so you don't see it unless you redirect it to a file or something.

Rob Kielty
  • 7,464
  • 7
  • 35
  • 50
Colin Burnett
  • 10,001
  • 6
  • 28
  • 40
60

That construct sends the standard error stream (stderr) to the current location of standard output (stdout) - this currency issue appears to have been neglected by the other answers.

You can redirect any output handle to another by using this method but it's most often used to channel stdout and stderr streams into a single stream for processing.

Some examples are:

# Look for ERROR string in both stdout and stderr.
foo 2>&1 | grep ERROR

# Run the less pager without stderr screwing up the output.
foo 2>&1 | less

# Send stdout/err to file (with append) and terminal.
foo 2>&1 |tee /dev/tty >>outfile

# Send stderr to normal location and stdout to file.
foo >outfile1 2>&1 >outfile2

Note that that last one will not direct stderr to outfile2 - it redirects it to what stdout was when the argument was encountered (outfile1) and then redirects stdout to outfile2.

This allows some pretty sophisticated trickery.

paxdiablo
  • 772,407
  • 210
  • 1,477
  • 1,841
  • 5
    Although that last example would be much clearer as: foo >outfile2 2>outfile1 – Michael Cramer May 04 '09 at 00:15
  • 4
    Clearer, yes, but that wouldn't show the "positional" nature of redirection. The example is contrived since it's not usually useful to do this in a single line - the method becomes really useful when different parties are responsible for the different parts of the redirection. For example, when a script does one bit of redirection and you run it with another bit. – paxdiablo May 04 '09 at 00:19
  • 6
    I just realized that the last example also resolves a long standing confusion I had regarding why this: `some_program 2>&1 > /dev/null` does not work like this: `some_program > /dev/null 2>&1`. – snapfractalpop Dec 20 '12 at 13:58
  • Your comment about the last example is worth its letters in gold :-) I never thought that these redirectional arguments are positional... I think this is pretty important to know. – Nils-o-mat Apr 18 '18 at 09:04
23

2 is the console standard error.

1 is the console standard output.

This is the standard Unix, and Windows also follows the POSIX.

E.g. when you run

perl test.pl 2>&1

the standard error is redirected to standard output, so you can see both outputs together:

perl test.pl > debug.log 2>&1

After execution, you can see all the output, including errors, in the debug.log.

perl test.pl 1>out.log 2>err.log

Then standard output goes to out.log, and standard error to err.log.

I suggest you to try to understand these.

Peter Mortensen
  • 28,342
  • 21
  • 95
  • 123
Marcus Thornton
  • 5,043
  • 5
  • 41
  • 45
  • The second sample is wrong: as order precedence *STDERR* is redirected to *STDOUT*, only default *STDOUT* will be written to *debug.log* (not *STDERR*) see [my answer](http://stackoverflow.com/a/16283739/1765658) (the paragraph #2)! To ensure **both** to be redirected to same file, you have to invert redirections directives: `perl test.pl > debug.log 2>&1` – F. Hauri Feb 10 '15 at 16:04
22

2>&1 is a POSIX shell construct. Here is a breakdown, token by token:


2: "Standard error" output file descriptor.

>&: Duplicate an Output File Descriptor operator (a variant of Output Redirection operator >). Given [x]>&[y], the file descriptor denoted by x is made to be a copy of the output file descriptor y.

1 "Standard output" output file descriptor.

The expression 2>&1 copies file descriptor 1 to location 2, so any output written to 2 ("standard error") in the execution environment goes to the same file originally described by 1 ("standard output").


Further explanation:

File Descriptor: "A per-process unique, non-negative integer used to identify an open file for the purpose of file access."

Standard output/error: Refer to the following note in the Redirection section of the shell documentation:

Open files are represented by decimal numbers starting with zero. The largest possible value is implementation-defined; however, all implementations shall support at least 0 to 9, inclusive, for use by the application. These numbers are called "file descriptors". The values 0, 1, and 2 have special meaning and conventional uses and are implied by certain redirection operations; they are referred to as standard input, standard output, and standard error, respectively. Programs usually take their input from standard input, and write output on standard output. Error messages are usually written on standard error. The redirection operators can be preceded by one or more digits (with no intervening characters allowed) to designate the file descriptor number.

wjordan
  • 17,115
  • 2
  • 72
  • 91
20

I found this very helpful if you are a beginner read this

Update:
In Linux or Unix System there are two places programs send output to: Standard output (stdout) and Standard Error (stderr).You can redirect these output to any file.

Like if you do this

ls -a > output.txt

Nothing will be printed in console all output (stdout) is redirected to output file.

And if you try print the content of any file that does not exits means output will be an error like if you print test.txt that not present in current directory

cat test.txt > error.txt

Output will be

cat: test.txt :No such file or directory

But error.txt file will be empty because we redirecting the stdout to a file not stderr.

so we need file descriptor( A file descriptor is nothing more that a positive integer that represents an open file. You can say descriptor is unique id of file) to tell shell which type of output we are sending to file .In Unix /Linux system 1 is for stdout and 2 for stderr.

so now if you do this

ls -a 1> output.txt means you are sending Standard output (stdout) to output.txt.

and if you do this

cat test.txt 2> error.txt means you are sending Standard Error (stderr) to error.txt .

&1 is used to reference the value of the file descriptor 1 (stdout).

Now to the point 2>&1 means “Redirect the stderr to the same place we are redirecting the stdout”

Now you can do this

cat maybefile.txt > output.txt 2>&1

both Standard output (stdout) and Standard Error (stderr) will redirected to output.txt.

Thanks to Ondrej K. for pointing out

kundan bora
  • 337
  • 3
  • 9
  • 1
    Link only answers are problematic. Link may become defunct rendering the answer useless. You should always include enough detail in the answer itself. – Ondrej K. Apr 01 '20 at 17:51
18

To answer your question: It takes any error output (normally sent to stderr) and writes it to standard output (stdout).

This is helpful with, for example 'more' when you need paging for all output. Some programs like printing usage information into stderr.

To help you remember

  • 1 = standard output (where programs print normal output)
  • 2 = standard error (where programs print errors)

"2>&1" simply points everything sent to stderr, to stdout instead.

I also recommend reading this post on error redirecting where this subject is covered in full detail.

Andrioid
  • 3,320
  • 4
  • 25
  • 30
11

From a programmer's point of view, it means precisely this:

dup2(1, 2);

See the man page.

Understanding that 2>&1 is a copy also explains why ...

command >file 2>&1

... is not the same as ...

command 2>&1 >file

The first will send both streams to file, whereas the second will send errors to stdout, and ordinary output into file.

ams
  • 22,492
  • 4
  • 47
  • 71
8

People, always remember paxdiablo's hint about the current location of the redirection target... It is important.

My personal mnemonic for the 2>&1 operator is this:

  • Think of & as meaning 'and' or 'add' (the character is an ampers-and, isn't it?)
  • So it becomes: 'redirect 2 (stderr) to where 1 (stdout) already/currently is and add both streams'.

The same mnemonic works for the other frequently used redirection too, 1>&2:

  • Think of & meaning and or add... (you get the idea about the ampersand, yes?)
  • So it becomes: 'redirect 1 (stdout) to where 2 (stderr) already/currently is and add both streams'.

And always remember: you have to read chains of redirections 'from the end', from right to left (not from left to right).

Kurt Pfeifle
  • 78,224
  • 20
  • 220
  • 319
7

Redirecting Input

Redirection of input causes the file whose name results from the expansion of word to be opened for reading on file descriptor n, or the standard input (file descriptor 0) if n is not specified.

The general format for redirecting input is:

[n]<word

Redirecting Output

Redirection of output causes the file whose name results from the expansion of word to be opened for writing on file descriptor n, or the standard output (file descriptor 1) if n is not specified. If the file does not exist it is created; if it does exist it is truncated to zero size.

The general format for redirecting output is:

[n]>word

Moving File Descriptors

The redirection operator,

[n]<&digit-

moves the file descriptor digit to file descriptor n, or the standard input (file descriptor 0) if n is not specified. digit is closed after being duplicated to n.

Similarly, the redirection operator

[n]>&digit-

moves the file descriptor digit to file descriptor n, or the standard output (file descriptor 1) if n is not specified.

Ref:

man bash

Type /^REDIRECT to locate to the redirection section, and learn more...

An online version is here: 3.6 Redirections

PS:

Lots of the time, man was the powerful tool to learn Linux.

Peter Mortensen
  • 28,342
  • 21
  • 95
  • 123
yurenchen
  • 921
  • 10
  • 12
6

Provided that /foo does not exist on your system and /tmp does…

$ ls -l /tmp /foo

will print the contents of /tmp and print an error message for /foo

$ ls -l /tmp /foo > /dev/null

will send the contents of /tmp to /dev/null and print an error message for /foo

$ ls -l /tmp /foo 1> /dev/null

will do exactly the same (note the 1)

$ ls -l /tmp /foo 2> /dev/null

will print the contents of /tmp and send the error message to /dev/null

$ ls -l /tmp /foo 1> /dev/null 2> /dev/null

will send both the listing as well as the error message to /dev/null

$ ls -l /tmp /foo > /dev/null 2> &1

is shorthand

Matijs
  • 3,088
  • 2
  • 24
  • 38
5

This is just like passing the error to the stdout or the terminal.

That is, cmd is not a command:

$cmd 2>filename
cat filename

command not found

The error is sent to the file like this:

2>&1

Standard error is sent to the terminal.

Peter Mortensen
  • 28,342
  • 21
  • 95
  • 123
Kalanidhi
  • 4,132
  • 21
  • 41
3

unix_commands 2>&1

This is used to print errors to the terminal.

The following illustrates the process

  • When errors are produced, they are written into the standard error memory address &2 "buffer", from which the standard error stream 2 references.
  • When output is produced, it is written into the standard output memory address &1 "buffer", from which the standard output stream 1 references.

So take the unix_commands standard error stream 2, and redirect > the stream (of errors) to the standard output memory address &1, so that they will be streamed to the terminal and printed.

tim-montague
  • 12,048
  • 2
  • 50
  • 40
1

0 for input, 1 for stdout and 2 for stderr.

One Tip: somecmd >1.txt 2>&1 is correct, while somecmd 2>&1 >1.txt is totally wrong with no effect!

ch271828n
  • 5,683
  • 3
  • 24
  • 44
0

Note that 1>&2 cannot be used interchangeably with 2>&1.

Imagine your command depends on piping, for example:
docker logs 1b3e97c49e39 2>&1 | grep "some log"
grepping will happen across both stderr and stdout since stderr is basically merged into stdout.

However, if you try:
docker logs 1b3e97c49e39 1>&2 | grep "some log",
grepping will not really search anywhere at all because Unix pipe is connecting processes via connecting stdout | stdin, and stdout in the second case was redirected to stderr in which Unix pipe has no interest.

yuranos
  • 5,211
  • 4
  • 42
  • 50