Shell脚本启动有多种方式,各种方式会加载不同的脚本,直接决定当前shell中的环境变量,具体shell启动方式如下:
1. 交互登录Shell启动,或者使用--login参数启动
登录Shell就是在输入用户名和密码登录后得到的Shell,比如从字符终端登录或者用telnet
/ssh
从远程登录。
这样启动bash
会自动执行以下脚本:
1).首先执行
/etc/profile
,系统中每个用户登录时都要执行这个脚本,如果系统管理员希望某个设置对所有用户都生效,可以写在这个脚本里2) . 然后依次查找当前用户主目录的
~/.bash_profile
、~/.bash_login
和~/.profile
三个文件,找到第一个存在并且可读的文件来执行,如果希望某个设置只对当前用户生效,可以写在这个脚本里,由于这个脚本在/etc/profile
之后执行,/etc/profile
设置的一些环境变量的值在这个脚本中可以修改,也就是说,当前用户的设置可以覆盖(Override)系统中全局的设置。~/.profile
这个启动脚本是sh
规定的,bash
规定首先查找以~/.bash_
开头的启动脚本,如果没有则执行~/.profile
,是为了和sh
保持一致。3) .顺便一提,在退出登录时会执行
~/.bash_logout
脚本(如果它存在的话)。
2. 交互非登录Shell启动
在图形界面下开一个终端窗口,或者在登录Shell提示符下再输入bash
命令,就得到一个交互非登录的Shell。
这种Shell在启动时自动执行~/.bashrc
脚本,并以fork子shell的方式执行,即继承父shell的环境变量,并依据.bashrc来修改,执行完后本地变量不会保存到父shell中去。
有时为了使登录Shell也能自动执行~/.bashrc
,通常在~/.bash_profile
中调用~/.bashrc
:
if [ -f ~/.bashrc ]; then
. ~/.bashrcfi
上面的意思是,如果~/.bashrc
文件存在则source
它。多数Linux发行版在创建帐户时会自动创建~/.bash_profile
和~/.bashrc
脚本,~/.bash_profile
中通常都有上面这几行。所以,如果要在启动脚本中做某些设置,使它在图形终端窗口和字符终端的Shell中都起作用,最好就是在~/.bashrc
中设置。
为什么登录Shell和非登录Shell的启动脚本要区分开呢?最初的设计是这样考虑的,如果从字符终端或者远程登录,那么登录Shell是该用户的所有其它进程的父进程,也是其它子Shell的父进程,所以环境变量在登录Shell的启动脚本里设置一次就可以自动带到其它非登录Shell里,而Shell的本地变量、函数、alias
等设置没有办法带到子Shell里,需要每次启动非登录Shell时设置一遍,所以就需要有非登录Shell的启动脚本,所以一般来说在~/.bash_profile
里设置环境变量,在~/.bashrc
里设置本地变量、函数、alias
等。如果你的Linux带有图形系统则不能这样设置,由于从图形界面的窗口管理器登录并不会产生登录Shell,所以环境变量也应该在~/.bashrc
里设置。
3. 非交互启动
为执行脚本而fork
出来的子Shell是非交互Shell,启动时执行的脚本文件由环境变量BASH_ENV
定义,相当于自动执行以下命令:
if [ -n "$BASH_ENV" ]; then
. "$BASH_ENV";
fi
如果环境变量BASH_ENV
的值不是空字符串,则把它的值当作启动脚本的文件名,source
这个脚本。
4. 以sh命令启动
如果以sh
命令启动bash
,bash
将模拟sh
的行为,以~/.bash_
开头的那些启动脚本就不认了。所以,如果作为交互登录Shell启动,或者使用--login参数启动,则依次执行以下脚本:
/etc/profile
~/.profile
如果作为交互Shell启动,相当于自动执行以下命令:
if [ -n "$ENV" ]; then . "$ENV"; fi
如果作为非交互Shell启动,则不执行任何启动脚本。通常我们写的Shell脚本都以#! /bin/sh
开头,都属于这种方式。