2

I am trying to better understand printf so I read multiple pages on this command, but I also stumbled upon different behavior of %q directive. Namely, stated on this page:

What is the difference between printf of bash shell, and /usr/bin/printf? The main difference is that the bash built-in printf supports the %q format specifier, which prints escape symbols alongside characters for safe shell input, while /usr/bin/printf does not support %q.


It behaves different, but I may not fully understand what the sentence means. The following is different, yet I think equally effective:

info printf | grep -A 4 '%q'

An additional directive %q, prints its argument string in a format that can be reused as input by most shells. Non-printable characters are escaped with the POSIX proposed $'' syntax, and shell metacharacters are quoted appropriately. This is an equivalent format to ls --quoting=shell-escape output.

$ touch 'printf testing %q.delete'
# no output

$ \ls -l printf\ testing\ %q.delete
-rw-rw-r-- 1 vlastimil vlastimil 0 Jun 15 23:21 'printf testing %q.delete'

$ /usr/bin/printf '%q\n' printf\ testing\ %q.delete
'printf testing %q.delete'

$ printf '%q\n' printf\ testing\ %q.delete  # Bash builtin
printf\ testing\ %q.delete

Can anyone elaborate on these differences, maybe adding in which scenarios it is advised to use this or that one? Thank you.

4
  • 1
    This question is similar to: Escape a variable for use as content of another script. If you believe it’s different, please edit the question, make it clear how it’s different and/or how the answers on that question are not helpful for your problem. Commented Jun 16 at 4:59
  • 1
    The differencees between the two styles is dealt with exhaustively in unix.stackexchange.com/a/600214/70524 Commented Jun 16 at 4:59
  • Do you mean the differences between ways of quoting the output, or differences in how / if the different printf implementations support %q? While Bash, zsh and GNU coreutils might support it, it doesn't guarantee every single shell or standalone printf implementation does. Commented Jun 16 at 8:54
  • 1
    @muru Wow, it makes you wonder how any software ever can run successfully ;-) Commented Jun 19 at 6:46

4 Answers 4

13

The page you link is outdated. /usr/bin/printf is generally provided by coreutils, and in coreutils v8.25 support was added for %q to printf:

printf now supports the '%q' format to print arguments in a form that is reusable by most shells, with non-printable characters escaped with the POSIX proposed $'...' syntax.

coreutils v8.25 was released 2016-01-20, so at this point most desktop/server Linux systems would have support.

2
  • 2
    "most [normal/whatever] Linux systems"..., well, the Linux systems using GNU userland anyway. Which might well be a majority of desktop and server systems, but embedded / small footprint systems (be it on small hardware or in containers) often run something like Busybox, which doesn't support %q. And Android is a rather common Linux-based OS, too. It uses toybox, and toybox's printf also doesn't seem to have %q. Commented Jun 19 at 17:27
  • @ilkkachu yeah, that's what I was trying to imply by "normal", but apparently that was too vague and also someone edited it out. Putting "desktop/server" there is an improvement. Explaining the specifics of coreutils alternatives would be a good fit for a separate answer, I think Commented Jun 21 at 15:41
10

The main difference between Bash built-in and external printf is that Bash supports -v to assign the result to a variable. A command obviously cannot change shell variables in its parent.

1
  • 2
    Also, the external printf (as well as /usr/local/bin/gprintf on FreeBSD systems with the coreutils package installed) supports parameter re-ordering, like /usr/bin/printf '%2$s, %3$s, and %1$s\n' 'the other' this that Commented Jun 16 at 17:35
6

Historically, each of these tools has been separately maintained (albeit with reference to standards and knowledge of each other). Assuming that we're referring to the /usr/bin/printf provided by Coreutils1, the functionality is quite similar, as you've seen.

So we have two sort-of independent implementations of the same general idea, each with a long version history. There is only one good general conclusion from this information, that doesn't care about whether your system is upgraded and can be expected to remain correct in the future:

The main difference between the Bash built-in printf and external printf is that the Bash built-in printf is a Bash built-in, while the external printf is an external executable.

Of course, this has consequences:

  • As Toby Speight says, the Bash builtin can implement shell-specific features (like the -v option) that are not possible for programs;

  • a Bash script using a builtin avoids the overhead of spawning the printf executable (on the order of milliseconds, while the builtin can complete on the order of microseconds);

  • the Bash builtin is only available in a Bash shell.

But these are, again, consequences of that general difference between builtins and programs. It has nothing in particular to do with printf.

Other shells may provide similar builtins. /usr/bin/printf is there as a fallback in case your shell doesn't have such a builtin. It's inherently slower (because a program has to start up) and harder to use explicitly (because you need to give the path name if you want to be sure of avoiding a builtin). But it offers greater (not perfect) compatibility - you can use it regardless of the shell, and it will work the same way regardless of the shell. Therefore, it's something you use when you need to.

1 Nowadays, of course, it normally would be that. But anyone with sudo rights can compile something, name it printf and stick it in /usr/bin. Any distro maintainer can provide something different. And more practically, future systems might use a printf implementation from e.g. uutils by default instead.

5

I think you're getting confused because of the statement "/usr/bin/printf does not support %q" in the first page you quoted. In fact, some versions of the external printf command do support the %q format specifier, but many do not. For example, on macOS, here's what I get when I try to use it:

$ /usr/bin/printf '%q\n' wibble
printf: illegal format character q

And on NetBSD:

$ /usr/bin/printf '%q\n' wibble
printf: %q: invalid directive

While on Linux, using the GNU coreutils version of /usr/bin/printf:

$ /usr/bin/printf '%q\n' wibble
wibble

You're apparently using the GNU version, so on your system it is supported (but it prefers slightly different quoting/escaping modes from bash's builtin).

Also, in your tests, you're including %q in the string (/filename) to be printed (i.e. printf testing %q.delete), but % sequences have no special meaning in those contexts; it's only in a printf format string that % introduces a format specifier.

1
  • 1
    Also e.g. on embedded systems, you might have Busybox, which doesn't seem to support %q (as configured on Ubuntu anyway). Not all shells have it either, e.g. Dash, the /bin/sh on Debian and Ubuntu, doesn't. Commented Jun 17 at 7:07

You must log in to answer this question.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.