Bash Alias and Functions – Comprehensive Guide

aliasbashfunction

From the bash manual

The rules concerning the definition and use of aliases are somewhat
confusing. Bash always reads at least one complete line of input
before executing any of the commands on that line. Aliases are
expanded when a command is read, not when it is executed. Therefore,
an alias definition appearing on the same line as another command does
not take effect until the next line of input is read. The commands
following the alias definition on that line are not affected by the
new alias. This behavior is also an issue when functions are executed.
Aliases are expanded when a function definition is read, not when the function is executed, because a function definition is itself a
compound command. As a consequence,
aliases defined in a function are not available until after that function is executed. To be safe, always put alias definitions on a
separate line, and do not use alias in compound commands.

The two sentences "Aliases are expanded when a function definition is read, not when the function is executed" and "aliases defined in a function are not available until after that function is executed" seem to be contrary to each other.

Can you explain what they mean respectively?

Best Answer

  1. Aliases are expanded when a function definition is read, not when the function is executed …

    $ echo "The quick brown fox jumps over the lazy dog." > myfile
     
    $ alias myalias=cat
     
    $ myfunc() {
    >     myalias myfile
    > }
     
    $ myfunc
    The quick brown fox jumps over the lazy dog.
     
    $ alias myalias="ls -l"
     
    $ myalias myfile
    -rw-r--r-- 1 myusername mygroup 45 Dec 13 07:07 myfile
     
    $ myfunc
    The quick brown fox jumps over the lazy dog.

    Even though myfunc was defined to call myalias, and I’ve redefined myalias, myfunc still executes the original definition of myalias.  Because the alias was expanded when the function was defined.  In fact, the shell no longer remembers that myfunc calls myalias; it knows only that myfunc calls cat:

    $ type myfunc
    myfunc is a function
    myfunc ()
    {
    cat myfile
    }
  2. … aliases defined in a function are not available until after that function is executed.

    $ echo "The quick brown fox jumps over the lazy dog." > myfile
     
    $ myfunc() {
    >     alias myalias=cat
    > }
     
    $ myalias myfile
    -bash: myalias: command not found
     
    $ myfunc
     
    $ myalias myfile
    The quick brown fox jumps over the lazy dog.

    The myalias alias isn’t available until the myfunc function has been executed.  (I believe it would be rather odd if defining the function that defines the alias was enough to cause the alias to be defined.)