53

In the Perl documentation, perlrun(1) suggests launching Perl scripts using a bilingual shell/Perl header:

#!/bin/sh
#! -*-perl-*-
eval 'exec perl -x -wS $0 ${1+"$@"}'
    if 0;

What does ${1+"$@"} mean? I tried using "$@" instead (using Bash as /bin/sh), and it seems to work just as well.


Edit

Two answers below say that it's supposed to be ${1:+"$@"}. I am aware of the ${parameter:+word} ("Use Alternate Value") syntax documented in bash(1). However, I am unconvinced, because

  1. Both ${1+"$@"} and "$@" work just fine, even when there are no parameters. If I create simple.sh as

    #!/bin/sh
    eval 'exec /usr/bin/perl -x -S -- $0 "$@"'
        if 0;
    #!perl
    use Data::Dumper;
    print Dumper(\@ARGV);
    

    and question.sh as

    #!/bin/sh
    eval 'exec /usr/bin/perl -x -S -- $0 ${1+"$@"}'
        if 0;
    #!perl
    use Data::Dumper;
    print Dumper(\@ARGV);
    

    I can get both to work identically:

    $ ./question.sh 
    $VAR1 = [];
    $ ./question.sh a
    $VAR1 = [
              'a'
            ];
    $ ./question.sh a 'b c'
    $VAR1 = [
              'a',
              'b c'
            ];
    $ ./question.sh ""
    $VAR1 = [
              ''
            ];
    $ ./simple.sh 
    $VAR1 = [];
    $ ./simple.sh a
    $VAR1 = [
              'a'
            ];
    $ ./simple.sh a 'b c'
    $VAR1 = [
              'a',
              'b c'
            ];
    $ ./simple.sh ""
    $VAR1 = [
              ''
            ];
    
  2. Other sources on the Internet also use ${1+"$@"}, including one hacker who seems to know what he's doing.

Perhaps ${parameter+word} is an undocumented alternate (or deprecated) syntax for ${parameter:+word}? Could someone confirm that hypothesis?

2
  • Looks like grawlixes Commented Mar 22, 2013 at 9:57
  • Pre-4.2 versions of Zsh do word splitting on ${1+"$@"} so some places switch to "$@" on zsh. example
    – 7efkvNEq
    Commented Dec 28, 2021 at 12:30

3 Answers 3

64

That's for compatibility with the Bourne shell. The Bourne shell was an old shell that was first released with Unix version 7 in 1979 and was still common until the mid 90s as /bin/sh on most commercial Unices.

It is the ancestor of most Bourne-like shells like ksh, bash or zsh.

It had a few awkward features many of which have been fixed in ksh and the other shells and the new standard specification of sh, one of which is this:

With the Bourne shell (at least those variants where it has not been fixed): "$@" expands to one empty argument if the list of positional parameters is empty ($# == 0) instead of no argument at all.

${var+something} expands to "something" unless $var is unset. It is clearly documented in all shells but hard to find in the bash documentation as you need to pay attention to this sentence:

When not performing substring expansion, using the forms documented below, bash tests for a parameter that is unset or null. Omitting the colon results in a test only for a parameter that is unset.

So ${1+"$@"} expands to "$@" only if $1 is set ($# > 0) which works around that limitation of the Bourne shell.

Note that the Bourne shell is the only shell with that problem. Modern shs (that is sh conforming to the POSIX specification of sh (which the Bourne shell is not)) don't have that issue. So you only need that if you need your code to work on very old systems where /bin/sh might be a Bourne shell instead of a standard shell (note that POSIX doesn't specify the location of the standard sh, so for instance on Solaris before Solaris 11, /bin/sh was still a Bourne shell (though did not have that particular issue) while the normal/standard sh was in another location (/usr/xpg4/bin/sh)).

There is a problem in that perlrun perldoc page in that $0 is not quoted though.

See http://www.in-ulm.de/~mascheck/various/bourne_args/ for more information.

4
  • Can't reproduce in Heirloom. Probably not applicable to Solaris.
    – ormaaj
    Commented Mar 19, 2013 at 21:18
  • 2
    @ormaaj. Yes, see the page I linked. It was fixed in SVR3, And Solaris/SunOS above 5 rebased on SVR4. And that page mentions that 4.1.x didn't have the issue either. Commented Mar 19, 2013 at 21:21
  • Ah I see. <3 Mascheck pages.
    – ormaaj
    Commented Mar 19, 2013 at 21:38
  • +1 for the thoroughness and definitely for the reference to the obscure sentence in the man page. Commented Sep 15, 2023 at 20:12
6

There is a difference between:

command ""

and

command

In one, you are passing one argument that is an empty string. In the second, there are zero arguments passed.

For both, "$@" will equate to the same thing: "". But using ${1:+"$@"} would be "" for the first and no arguments passed for the second, which was the intent.

This becomes important if you are doing something the script below, called sshwrapper, which you call with an optional command or no arguments to get an interactive shell.

: sshwrapper [command]
exec /usr/bin/ssh "${HOSTNAME:-localhost}" "$@"

This would attempt to run "" on the remote host (which just returns), it would not be an interactive shell.

: sshwrapper [command]
exec /usr/bin/ssh "${HOSTNAME:-localhost}" ${1:+"$@"}

Would start an interactive shell on the remote host because the exec would correctly interpret the variable as nothing, not an empty string.

Read up on the usage of "${parameter:+word}" ("Use Alternate Value") and similar variable expansion strings in the bash manual pages.

2
  • this seems to apply only to Bourne shell and not for any POSIX-conformant sh implementation (see other answer). Commented Jan 7, 2015 at 8:08
  • 4
    No, ${1:+"$@"} would expand to no argument if $1 was empty. You want ${1+"$@"}. Basically you have it reversed. Commented Jan 7, 2015 at 9:40
2

I summarized Stéphane Chazelas' answer:

  • ${1:+"$@"}' test if $1 null or unset
  • ${1+"$@"}' test if $1 unset

so if use second one with parameter "", that means $1 is null, but it doesn't test whether it is null or not, it just see it have been already setted nevertheless it it empty or not, so it will expand $@, but you use ${1:+"$@"}' with "", it will not expand $@ any more.

1
  • 7
    This certainly summarized, although it might not clarify ...
    – Archemar
    Commented Mar 11, 2015 at 9:49

You must log in to answer this question.

Not the answer you're looking for? Browse other questions tagged .