如果我对.bashrc进行了更改,如何在不注销并重新登录的情况下重新加载它?


当前回答

为了补充和对比两个最流行的答案,.~/。bashrc和exec bash:

两种解决方案都有效地重新加载~/.bashrc,但存在差异:

./.bashrc或source~/.bashrc将保留当前shell会话:除了将~/.bashrc重新加载到当前shell(sourcing)中所做的修改外,当前shell进程及其状态将被保留,其中包括环境变量、shell变量、shell选项、shell函数和命令历史记录。exec bash,或者更有力地说,exec“$bash”[1],将用新实例替换当前shell,因此只保留当前shell的环境变量(包括您在会话中临时定义的环境变量)。换句话说:在shell变量、shell函数、shell选项和命令历史记录方面对当前shell的任何特殊更改都将丢失。

根据您的需要,可以选择一种或另一种方法。


注:上述内容同样适用于其他外壳:

要将exec方法应用于任何默认shell,请使用exec$shell同样,源方法要求您知道并指定特定于shell的初始化文件的名称;例如,对于zsh:.~/。zshrc公司


[1] 理论上,execbash可以执行一个不同于启动当前shell的bash可执行文件,如果它恰好存在于$PATH前面列出的目录中。由于特殊变量$BASH始终包含启动当前shell的可执行文件的完整路径,因此exec“$BASH”保证使用相同的可执行程序。注意$BASH附近的“…”:双引号确保变量值按原样使用,而无需BASH解释;如果该值没有嵌入空格或其他shell元字符(在本例中很可能是这样),则不必严格使用双引号,但使用它们是一个好习惯。

其他回答

或者您可以使用:

exec bash

这样做也是一样的,而且更容易记住(至少对我来说)。

exec命令通过运行指定的命令行完全替换shell进程。在我们的示例中,它用一个新的bash实例(用更新的配置文件)替换当前的shell。

我知道你想要一个shell,因为在注销并再次登录后。我认为实现这一目标的最佳方式是:

exec env -i HOME="$HOME" "$SHELL" -l

exec将替换当前的shell,这样当新的shell退出时就不会剩下它了。env将创建一个新的空环境,使用-i,我们添加$HOME,这样$shell提供的shell(通常是bash)可以找到~/.profile/~/.bash_profile(因此(在ubuntu上或如果指定)~/.bash rc)。不过我不完全确定。

为了补充和对比两个最流行的答案,.~/。bashrc和exec bash:

两种解决方案都有效地重新加载~/.bashrc,但存在差异:

./.bashrc或source~/.bashrc将保留当前shell会话:除了将~/.bashrc重新加载到当前shell(sourcing)中所做的修改外,当前shell进程及其状态将被保留,其中包括环境变量、shell变量、shell选项、shell函数和命令历史记录。exec bash,或者更有力地说,exec“$bash”[1],将用新实例替换当前shell,因此只保留当前shell的环境变量(包括您在会话中临时定义的环境变量)。换句话说:在shell变量、shell函数、shell选项和命令历史记录方面对当前shell的任何特殊更改都将丢失。

根据您的需要,可以选择一种或另一种方法。


注:上述内容同样适用于其他外壳:

要将exec方法应用于任何默认shell,请使用exec$shell同样,源方法要求您知道并指定特定于shell的初始化文件的名称;例如,对于zsh:.~/。zshrc公司


[1] 理论上,execbash可以执行一个不同于启动当前shell的bash可执行文件,如果它恰好存在于$PATH前面列出的目录中。由于特殊变量$BASH始终包含启动当前shell的可执行文件的完整路径,因此exec“$BASH”保证使用相同的可执行程序。注意$BASH附近的“…”:双引号确保变量值按原样使用,而无需BASH解释;如果该值没有嵌入空格或其他shell元字符(在本例中很可能是这样),则不必严格使用双引号,但使用它们是一个好习惯。

请注意,$SHELL可能会产生意想不到的结果例如,连接到Docker环境

echo $SHELL
/usr/sbin/nologin

所以,如果你尝试,你就会断开连接

exec $SHELL
This account is currently not available.

所以你可能需要使用更复杂的东西,比如

exec $(pgrep -l sh | grep "^`echo $$` " | cut -d" " -f2)

假设每个shell都包含“sh”

pgrep -l sh | grep "^`echo $$` " | cut -d" " -f2

生成完整的命令如果有参数或标志,则可能必须使用-f2、3、4或尝试

pgrep -l sh | grep "^`echo $$` " | sed -E 's/^[0-9]+ //'

但请阅读以上所有建议就我个人而言,我不喜欢泄露历史。。。但这取决于你和你的需要

我编写了一组名为bash_magic的脚本,这些脚本可以跨多个shell自动执行此过程。如果您更新bashmagicshell目录(默认情况下为.bash.d)中的shell文件,它将在下一个提示下自动源代码更新。因此,一旦您进行了更改,只需按Enter键/回车键,任何更新都将被获取。