38
( du /etc; du /var; ) > tmp.txt { du /etc; du /var; } > tmp.txt 

Is there a difference between the () and {}?

The output of tmp.txt seems exactly the same, and I was wondering whether i'm missing something here.

5
  • 1
    See Grouping Commands under 2.9.4 Compound Commands Commented Mar 4, 2016 at 17:10
  • 3
    try inserting exit ; between the du. Commented Mar 4, 2016 at 17:10
  • This is the closest answer I found in the web. The exact issue has not been explained in this site. Commented Mar 4, 2016 at 18:30
  • 2
    Both of them are Grouping Commands, but { list; } executes commands in current shell environment. Commented Mar 4, 2016 at 18:31
  • @Archemar Interesting that if pipe is used the current shell will not be closed. Example: ( echo 1; exit; echo 2; ) | less. In less will be shown 1. Checked in bash shell. Commented Apr 9, 2024 at 12:56

1 Answer 1

51

Parentheses cause the commands to be run in a subshell.

Braces cause the commands to be grouped together but not in a subshell.

Given that your example does not use side-effects, there is no real difference between both. If there were side-effects, e.g. setting or modifying shell variables, there is a difference as such side-effects applied to a sub-shell will be forgotten when this sub-shell ends.

To understand the "side-effect", see the following examples:

Using parentheses ():

v="test"; ( echo $v; v="modified"; echo $v; ); echo $v; # output: # test # modified # test 

Using curly braces {}:

v="test"; { echo $v; v="modified"; echo $v; }; echo $v; # output: # test # modified # modified 

If you however take a closer look and compare the behavior of different shell implementations, it becomes confusing:

The Bourne Shell e.g. runs grouped commands in a subshell in case there is an I/O redirection and ksh93 avoids subshells by implementing virtual subshell behavior that is done by creating a temporary copy of new parameters. Whether this is always 100% correct is not known, ksh93 Version M 1993-12-28 s+ from 2009 e.g. implements $(...) incorrectly and $(alias a=b) affects the main shell.

So in general: if you are interested in specific aspects, be careful and check your shell for it's actual behavior.

5
  • 1
    (I wouldn't have bothered mentioning it but since you're maintaining a Bourne shell ;)), ...not in a subshell, except for the Bourne shell that runs compound commands in subshells when redirected. Commented Mar 4, 2016 at 17:11
  • Well parentheses grant the commands to be in a subshell, with braces, it still may happen ;-) but ksh93 does not even create a real subshell for parentheses, but rather emulates the effects of a subshell by using a temporary new copy of parameters. Commented Mar 4, 2016 at 17:14
  • Yes, you'll notice that the POSIX spec is careful to talk only of subshell environment with no implication that it may involve a child process. And with many shells that implement subshells using a child process, (a;b) will not spawn more processes than {a;b;} (if b is not a builtin nor function nor compound command and there's no local trap) as b will be executed in that child subshell process. Commented Mar 4, 2016 at 17:20
  • So if I understand correctly, the only difference is that: parentheses use a subshell and braces don't? And the output stays the same, regardless of the subshell (in this case). Thankyou, this helped! Commented Mar 4, 2016 at 17:32
  • See my new paragraph about side-effects. In your example, there is no visible difference. Commented Mar 4, 2016 at 17:36

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.