Sourcing a script will run the commands in the current shell process.
Executing a script will run the commands in a new shell process.
Use source if you want the script to change the environment in your currently running shell. use execute otherwise.
If you are still confused, please read on.
Terminology
To clarify some common confusion about the syntax to execute and the syntax to source:
./myscript
This will execute myscript provided that the file is executable and located in the current directory. The leading dot and slash (./) denotes the current directory. This is necessary because the current directory is usually not (and usually should not be) in $PATH.
myscript
This will execute myscript if the file is executable and located in some directory in $PATH.
source myscript
This will source myscript. The file need not be executable but it must be a valid shell script. The file can be in current directory or in a directory in $PATH.
. myscript
This will also source myscript. This "spelling" is the official one as defined by POSIX. Bash defined source as an alias to the dot.
Demonstration
Consider myscript.sh with following content:
!/bin/sh
demonstrate setting a variable
echo "foo: "$(env | grep FOO)
export FOO=foo
echo "foo: "$(env | grep FOO)
demonstrate changing of working directory
echo "PWD: "$PWD
cd somedir
echo "PWD: "$PWD
Before we execute the script first we check the current environment:
$ env | grep FOO
$ echo $PWD
/home/lesmana
The variable FOO is not defined and we are in the home directory.
Now we execute the file:
$ ./myscript.sh
foo:
foo: FOO=foo
PWD: /home/lesmana
PWD: /home/lesmana/somedir
Check the environment again:
$ env | grep FOO
$ echo $PWD
/home/lesmana
The variable FOO is not set and the working directory did not change.
The script output clearly shows that the variable was set and the directory was changed. The check afterwards show that the variable is not set and the directory not changed. What happened? The changes were made in a new shell. The current shell spawned a new shell to run the script. The script is running in the new shell and all changes to the environment take effect in the new shell. After the script is done the new shell is destroyed. All changes to the environment in the new shell are destroyed with the new shell. Only the output text is printed in the current shell.
Now we source the file:
$ source myscript.sh
foo:
foo: FOO=foo
PWD: /home/lesmana
PWD: /home/lesmana/somedir
Check the environment again:
$ env | grep FOO
FOO=foo
$ echo $PWD
/home/lesmana/somedir
The variable FOO is set and the working directory has changed.
Sourcing the script does not create a new shell. All commands are run in the current shell and changes to the environment take effect in the current shell.
Note that in this simple example the output of executing is the same as sourcing the script. This is not necessarily always the case.
Another Demonstration
Consider following script pid.sh:
!/bin/sh
echo $$
(the special variable $$ expands to the PID of the current running shell process)
First print the PID of the current shell:
$ echo $$
25009
Source the script:
$ source pid.sh
25009
Execute the script, note the PID:
$ ./pid.sh
25011
Source again:
$ source pid.sh
25009
Execute again:
$ ./pid.sh
25013
You can see that sourcing the script runs in the same process while executing the script creates a new process everytime. That new process is the new shell which was created for the execution of the script. Sourcing the script does not create a new shell and thus the PID stays the same.
Summary
Both sourcing and executing the script will run the commands in the script line by line, as if you typed those commands by hand line by line.
The differences are:
When you execute the script you are opening a new shell, type the commands in the new shell, copy the output back to your current shell, then close the new shell. Any changes to environment will take effect only in the new shell and will be lost once the new shell is closed.
When you source the script you are typing the commands in your current shell. Any changes to the environment will take effect and stay in your current shell.
Use source if you want the script to change the environment in your currently running shell. use execute otherwise.
See also:
https://stackoverflow.com/questions/6331075/why-do-you-need-dot-slash-before-script-name-to-run-it-in-bash
https://askubuntu.com/questions/182012/is-there-a-difference-between-and-source-in-bash-after-all
==============================================
src:
https://superuser.com/questions/176783/what-is-the-difference-between-executing-a-bash-script-vs-sourcing-it
Top comments (0)