遇到的问题:设置Mac的.bash_profile文件没有用,每次登陆的时候还是要source .bash_profile。
1、前言:Shell脚本的类型
Shell是一个用户接口,是一个命令解释器,除此之外,shell命令本身还可以作为程序设计语言,将多个shell命令组合起来,编写能实现系统或用户所需功能的程序。Shell类型:bash,bsh,csh,tcsh,ksh,ash,zsh。
每次登陆终端的时候,要注意一下自己用的是什么shell,Linux很多时候是bash(默认),Mac很多时候用的是zsh,可以用echo $SHELL看一下,不同的shell会读取不同的文件。
shell 也可分为 login shell 和 non-login shell;也可以分为 interactive shell 和 non-interactive shell。
login shell 是用户登陆时,输入用户名和密码后启动的 shell,non-login shell 是登录以后所打开的 shell。interactive shell 在终端上执行,shell 等待你的输入,并且立即执行你提交的命令,跟用户存在交互;non-interactive shell 以 shell script(非交互)方式执行。比如:通过正常的 ssh 登录到服务器(输入用户名和密码)进入的 shell是login shell。但是进入 login shell 以后,再打开 GUI terminal 或者执行 bash 命令,此时打开的是 non-login shell。
但是,在 Mac OS X 中打开 iTerm.app 或者 Terminal.app 启动的 shell 是什么类型呢?通常来说,应该是 interactive, non-login shell,但实际上却是 interactive, login shell,至于为什么这样就不深究了。下面的测试代码可以证明:
[[ -o login ]] && echo 'yes' || echo 'no'
[[ -o interactive ]] && echo 'yes' || echo 'no'
2、环境变量是什么?
定义:环境变量是在操作系统中一个具有特定名字的对象,它包含了一个或者多个应用程序所将使用到的信息。
例如:Windows和DOS操作系统中的PATH环境变量,当要求系统运行一个程序而没有告诉它程序所在的完整路径时,系统除了在当前目录下面寻找此程序外,还应到PATH中指定的路径去找。用户通过设置环境变量,来更好的运行进程。
又比如:Golang将GOBIN添加到PATH变量中,就可以直接执行golang编译出的可执行文件,而不用非要到可执行文件的目录下执行。
简单可以理解为,环境变量就是一些被指定的文件夹路径,目的是为了更快速方便的找到想要的文件和文件夹。
参考:环境变量
3、profile、bashrc、bash_profile等等---bash
3.1 profile文件
profile(/etc/profile),用于设置系统级的环境变量和启动程序,在这个文件下配置会对所有用户生效。当用户登录(login)时,文件会被执行,并从/etc/profile.d目录的配置文件中查找shell设置。但是,不建议在/etc/profile文件中添加环境变量,因为在这个文件中添加的设置会对所有用户起作用。(Mac下的zsh不执行)
3.2 bashrc文件
这个文件用于配置函数或别名。bashrc文件有两种级别:系统级的位于/etc/bashrc、用户级的~/.bashrc,两者分别会对所有用户和当前用户生效。此类文件一般读取不读取主要看有没有
bashrc文件只会对指定的shell类型起作用,bashrc只会被bash shell调用。同理,zrc只会被zsh调用。(这就是为什么我在.bash_profile文件中添加PATH环境变量没用的原因)
3.3 bash_profile文件
bash_profile只有单一用户有效,文件存储位于~/.bash_profile,该文件是一个用户级的设置,可以理解为某一个用户的profile目录下。这个文件同样也可以用于配置环境变量和启动程序,但只针对单个用户有效。
和profile文件类似,bash_profile也会在用户登录(login)时生效,也可以用于设置环境变理。但与profile不同,bash_profile只会对当前用户生效。
参考:profile、bash_profile、bashrc文件的作用与区别
4、bash shell的读取顺序
读取顺序:
采用 interactive login shell 登录的时候,首先读取并执行 /etc/profile 文件中命令,然后再依次顺序寻找 ~/.bash_profile,~/.bash_login 和 ~/.profile 文件,然后从第一个存在且可读取的文件中读取并执行指令
如果是 interactive non-login shell, 启动时候,首先读取执行 /etc/bashrc,然后再读取执行 ~/.bashrc
参考:为什么 bash_profile 里面的设置没有生效
5、Mac下一般使用的zsh的读取顺序
zsh 启动过程中会依次读取以下文件:
1、/etc/zshenv
2、$ZDOTDIR/.zshenv($ZDOTDIR 未设置时默认为 $HOME,可以echo $HOME看一下,一般都是“/Users/用户”)
3、如果是 login shell,读取 /etc/zprofile, $ZDOTDIR/.zprofile
4、如果是 interactive shell,读取 /etc/zshrc, $ZDOTDIR/.zshrc
5、如果是 login shell,读取 /etc/zlogin, $ZDOTDIR/.zlogin
参考:【转】Mac OS X 中 Zsh 下 PATH 环境变量的正确设置
***:我的问题所在
我是用的是iTerm2.app,shell用的是zsh(一直都没有注意到这一点......所以大家一定要看一下自己用的shell类型),设置PATH环境变量却在~/.bash_profile 中设置,zsh是不会读取这个文件的。解决办法:方法很多,比如在上述~/.zprofile中添加PATH,又或者在~/.zrc中source ~/.bash_profile 等等。