How to format a number in bash, possibly with printf?

前端 未结 4 1781
遇见更好的自我
遇见更好的自我 2021-01-15 07:12

This so simple script is failing :

#!/bin/bash
n=1
printf -v fn \"%05d\" $n
echo $fn

with

3: printf: Illegal option -v
         


        
4条回答
  •  夕颜
    夕颜 (楼主)
    2021-01-15 07:31

    Per @Joe, this appears to be a duplicate of Whats the difference between running a shell script as ./script.sh and sh script.sh.

    Per @Telemachus, Debian and its derivatives use dash as their default shell. See http://wiki.ubuntu.com/DashAsBinSh for more information.

    Here's what I see on an Ubuntu system:

    $ ls -lF `which sh`
    lrwxrwxrwx 1 root root 4 Aug 15  2012 /bin/sh -> dash*
    $ ls -lF `which bash`
    -rwxr-xr-x 1 root root 959168 Mar 30  2013 /bin/bash*
    

    That explains why I was not able to reproduce the issue on Mac OS X 10.8.5. I did reproduce it on Ubuntu by invoking the script with sh instead of bash.

    I'm leaving the rest of my answer in place since it demonstrates some steps you might take to troubleshoot the problem.

    Can you check your version of bash?

    $ bash --version
    bash --version
    GNU bash, version 3.2.48(1)-release (x86_64-apple-darwin12)
    

    Does this even work?

    #!/bin/bash
    n=1
    printf -v fn "%05d" $n
    echo $fn
    

    Check the type of the name printf?

    $ type printf
    printf is a shell builtin
    $ function printf { echo "giggle" ; }
    giggle
    $ type printf
    printf is a function
    printf () 
    { 
        echo "giggle"
    }
    giggle
    $ 
    

    Check the builtin help for the printf builtin?

    $ help printf
    help printf
    
    printf: printf [-v var] format [arguments]
        printf formats and prints ARGUMENTS under control of the FORMAT. FORMAT
        is a character string which contains three types of objects: plain
        characters, which are simply copied to standard output, character escape
        sequences which are converted and copied to the standard output, and
        format specifications, each of which causes printing of the next successive
        argument.  In addition to the standard printf(1) formats, %b means to
        expand backslash escape sequences in the corresponding argument, and %q
        means to quote the argument in a way that can be reused as shell input.
        If the -v option is supplied, the output is placed into the value of the
        shell variable VAR rather than being sent to the standard output.
    

    Has the builtin printf been replaced by a definition from somewhere else? Here's a function I use for checking the definition of names in the shell:

    list () 
    { 
        if [[ 0 == $# ]]; then
            Log "";
            Log "FUNCTIONS:";
            Log "----------";
            declare -F;
            Log "";
            Log "EXPORTS:";
            Log "--------";
            export -p;
            Log "";
            Log "PRINTENV:";
            Log "--------";
            printenv;
        else
            while [[ ! -z "$1" ]]; do
                local name="$1";
                shift;
                if ! alias "${name}" 2> /dev/null; then
                    if ! declare -f "${name}"; then
                        if ! help "${name}" 2> /dev/null; then
                            if ! which "${name}"; then
                                Log "Not found: '${name}'";
                            fi;
                        fi;
                    fi;
                fi;
            done;
        fi
    }
    

    Here's the output when I run this in a fresh shell:

    $ list printf
    printf: printf [-v var] format [arguments]
        printf formats and prints ARGUMENTS under control of the FORMAT. FORMAT
        [… snip …]
    

    But if I redefine printf, it will show the definition:

    $ function printf { echo "kibble" ; }
    kibble
    $ printf
    kibble
    kibble
    $ list printf
    printf () 
    { 
        echo "kibble"
    }
    kibble
    $ 
    

    I am curious to hear what's really going on here!!!

    I like the other answer's suggestion to try invoking the script explicitly w/ bash:

    $ bash myscript.sh
    

    Here's what I see on an Ubuntu server:

    $ uname -a
    Linux rack 3.11.0-17-generic #31-Ubuntu SMP Mon Feb 3 21:52:43 UTC 2014 x86_64 x86_64 x86_64 GNU/Linux
    $ cat > dme.sh
    #!/bin/bash
    n=1
    printf -v fn "%05d" $n
    echo $fn
    $ chmod +x ./dme.sh
    $ ./dme.sh
    00001
    $ bash dme.sh
    00001
    $ sh dme.sh
    dme.sh: 3: printf: Illegal option -v
    

提交回复
热议问题