当前位置: 首页 > news >正文

Linux:bash在被调用时会读取哪些启动文件?

(本文基于5.1-6ubuntu1.1版本的bash)

bash在被调用时会读取哪些启动文件?要回答这个问题,首先要弄清楚两个概念:login shell和interactive shell。

login shell

login shell是指这样的shell:

  1. 第一个命令行参数(进程名)以-开头,也就是满足argv[0][0] == '-'
  2. 或者 设置了--login/-l选项

对于条件1,有一个例子:当执行su - [USER_NAME]时,su命令会启动一个login shell,argv[0][0]会被设置为-su手册中写到:

-, -l, --loginStart the shell as a login shell with an environment similar to a real login:•   clears all the environment variables except TERM and variables specified by --whitelist-environment•   initializes the environment variables HOME, SHELL, USER, LOGNAME, and PATH•   changes to the target user’s home directory•   sets argv[0] of the shell to '-' in order to make the shell a login shell

不满足login shell条件的即为non-login shell。

interactive shell

interactive shell是指这样的shell:

  1. 满足以下条件
    1. 没有非选项参数(如果设置了-s选项则可以有非选项参数(non-option arguments))
    2. 并且 没有设置-c选项
    3. 并且 标准输入和标准错误都被连接到终端
  2. 或者 设置了-i选项

对于一个interactive shell,PS1环境变量会被设置,同时$-中会含有i选项:

$ echo $PS1
\[\e]0;\u@\h: \w\a\]${debian_chroot:+($debian_chroot)}\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\w\[\033[00m\]\$
$ echo $-
himBHs

interactive shell,顾名思义,指可交互的shell。所以像是创建子进程执行一个脚本(bash ./foo.sh)或是执行一条命令(bash -c 'ls')时,创建出来的bash进程就显然不是可交互的,因此不属于interactive shell。除此之外,bash还把重定向了标准输入或者标准错误的bash排除在了interactive shell之外。所以对于bash 2>/dev/null这种情况,尽管实际上是可交互的,但仍然不属于interactive shell。下面这些命令创建的bash进程都不属于interactive shell:

$ bash ./foo.sh
$ bash -c 'ls'
$ bash < foo.sh
$ bash 2>/dev/null

然而,bash提供了一个选项-i,可以强行将一个bash设置为interactive shell,上面提供的4个例子中,只要加上了-i选项,它们就全部变成了interactive shell。

所以说,判断一个bash是否为interactive shell,不能从表现上看,还是要看它在被调用时设置了哪些参数。

在代码层面上,interactive shell的判断逻辑如下:

  /* First, let the outside world know about our interactive status.A shell is interactive if the `-i' flag was given, or if all ofthe following conditions are met:no -c commandno arguments remaining or the -s flag givenstandard input is a terminalstandard error is a terminalRefer to Posix.2, the description of the `sh' utility. */if (forced_interactive ||		/* -i flag */(!command_execution_string &&	/* No -c command and ... */wordexp_only == 0 &&		/* No --wordexp and ... */((arg_index == argc) ||		/*   no remaining args or... */read_from_stdin) &&		/*   -s flag with args, and */isatty (fileno (stdin)) &&	/* Input is a terminal and */isatty (fileno (stderr))))	/* error output is a terminal. */init_interactive ();elseinit_noninteractive ();

不满足interactive shell条件的即为non-interactive shell。

不同情况下读取的启动文件

在了解了login shell和interactive shell的概念后,我们就可以看看在不同情况下bash会读取哪些启动文件了。

  1. login && (interactive || non-interactive) shell
    1. 读取/etc/profile
    2. 读取~/.bash_profile~/.bash_login~/.profile中存在且可读的第一个
  2. non-login && interactive shell
    1. 读取/etc/bash.bashrc
    2. 读取~/.bashrc
  3. non-login && non-interactive shell
    1. 读取BASH_ENV环境变量中保存的文件路径,如果存在的话

需要注意的是,尽管login shell不会直接读取/etc/bash.bashrc或是~/.bashrc,但是通常会在满足条件时,通过/etc/profile~/.bash_profile/~/.bash_login/~/.profile间接读取这两个文件:

# /etc/profileif [ "${PS1-}" ]; thenif [ "${BASH-}" ] && [ "$BASH" != "/bin/sh" ]; then# The file bash.bashrc already sets the default PS1.# PS1='\h:\w\$ 'if [ -f /etc/bash.bashrc ]; then. /etc/bash.bashrcfielseif [ "$(id -u)" -eq 0 ]; thenPS1='# 'elsePS1='$ 'fifi
fi
# ~/.profileif [ -n "$BASH_VERSION" ]; then# include .bashrc if it existsif [ -f "$HOME/.bashrc" ]; then. "$HOME/.bashrc"fi
fi

特殊情况下读取的启动文件

sh的名字调用时

在这种情况下,bash会模仿sh的行为:

  1. login && (interactive || non-interactive) shell
    1. 读取/etc/profile
    2. 读取~/.profile
  2. non-login && interactive shell
    1. 读取ENV环境变量中保存的文件路径
  3. non-login && non-interactive shell
    1. 不读取启动文件

posix模式调用时

如果bash调用时设置了--posix参数,那么在interactive shell的情况下,bash会尝试从ENV环境变量中获取启动文件路径,不会读取其他启动文件。

ssh调用时

在被ssh调用时,或者标准输入是一个网络连接时,会触发一些特殊逻辑:在这种情况下,即使是non-login && non-interactive shell情况的组合,bash仍然会读取/etc/bash.bashrc~/.bashrc

  /* get the rshd/sshd case out of the way first. */if (interactive_shell == 0 && no_rc == 0 && login_shell == 0 &&act_like_sh == 0 && command_execution_string){run_by_ssh = (find_variable ("SSH_CLIENT") != (SHELL_VAR *)0) ||(find_variable ("SSH2_CLIENT") != (SHELL_VAR *)0);/* If we were run by sshd or we think we were run by rshd, execute~/.bashrc if we are a top-level shell. */if ((run_by_ssh || isnetconn (fileno (stdin))) && shell_level < 2){maybe_execute_file (SYS_BASHRC, 1);maybe_execute_file (bashrc_file, 1);return;}}
http://www.lryc.cn/news/408695.html

相关文章:

  • 帆软FineReport之替换函数
  • Redis的应用场景及类型
  • 【图像处理】不智能的目标识别
  • 《500 Lines or Less》(5)异步爬虫
  • Transformer!自注意力机制的高层级理解Attention Is All You Need!
  • 关于使用Postman在请求https网址没有响应,但是用浏览器有响应的问题解决
  • 【React 】开发环境搭建详细指南
  • 结构体笔记
  • Elasticsearch:Golang ECS 日志记录 - zerolog
  • Ip2region - 基于xdb离线库的Java IP查询工具提供给脚本调用
  • 研发管理革命:探索顶尖的工时系统选择
  • 微服务-MybatisPlus下
  • 【python_将一个列表中的几个字典改成二维列表,并删除不需要的列】
  • IDEA的pom.xml显示ignored 的解决办法
  • 2. 卷积神经网络无法绕开的神——LeNet
  • 【区块链】JavaScript连接web3钱包,实现测试网络中的 Sepolia ETH余额查询、转账功能
  • 关于珞石机器人二次开发SDK的posture函数的算法RX RY RZ纠正 C#
  • 【Three.js基础学习】17.imported-models
  • Spring Bean - xml 配置文件创建对象
  • uniapp map组件自定义markers标记点
  • Windows:批处理脚本学习
  • Dav_笔记10:Using SQL Plan Management之4
  • 通过json传递请求参数,如何处理动态参数和接口依赖
  • [240727] Qt Creator 14 发布 | AMD 推迟 Ryzen 9000芯片发布
  • PLSQL Developer工具查询数据,报错(动态性能表不可访问)
  • 基于 HTML+ECharts 实现智慧交通数据可视化大屏(含源码)
  • 探索 IT 领域的新宠儿:量子计算
  • TSPNet代码分析
  • Ubuntu上安装anaconda创建虚拟环境(各种踩坑版)
  • DC-5靶机通关