用python去修改linux环境变量为何无效?

我用python写了一个安装脚本,安装完了后想永久配置环境,我选择修改/etc/profile,然后修改完了后执行os.system("source …
关注者
45
被浏览
34,880

3 个回答

@a cook 很抱歉,不知为何,我没有收到你补充问题的通知,你也没有留言追问,所以我一直不知道有问题补充,一年后的现在才刚刚看到,不知道你的问题是否已经解决。

对你的需求,我的理解是是这样的,你在同一个python脚本中希望按顺序做这三件事:

  • 安装nasm
  • 将nasm的路径加入PATH中
  • 执行另一个软件包的./configure命令

如果我的理解没错的话,你可以这样做:

import os
os.environ['SOME_ENV_NAME'] = 'SOME_VALUE'
os.system('env')

env 是一个外部命令,作用是打印这个进程的所有的环境变量。os.environ 是 python 程序中用来访问自己的环境变量的方式。当python中向os.environ中添加一个新的变量时,python后续启动的子进程也会继承这个变量。所以,上面的代码,env命令打印的输出中,可以找到 SOME_ENV_NAME。

再举一个例子,如果你希望在PATH中添加一个路径:

import os
os.environ['PATH'] = '%s:%s' % (os.environ['PATH'], '/some/path/to/add')
os.system('some command')

其实,在Python中还可以有更“精确”的控制子进程环境变量的方式,使用 subprocess 下面的函数,如Popen()、call()、run() 等函数启动子进程时,都可以提供一个 env 变量,env变量是一个字典类型,你可以按需任意构造。具体可以参考文档:17.5. subprocess - Subprocess management - Python 3.6.4 documentation

最后,再次注意,这里所谓的修改环境变量,仅仅影响当前python进程后续启动的子进程,对当前python进程以外的环境是没有任何影响的。

---------- 8< ----------

以下是原回答

---------- 8< ----------

提问者可能没有正确的理解“环境变量”的含义,所以我先简单解释一下环境变量。

环境变量可以看作是进程的一些元信息,脱离进程谈环境变量是没有意义的。在创建进程时,可以给新创建的进程设置一些环境变量,而进程一旦运行起来之后,就没有常规的手段可以从外部修改这个进程的环境变量了。一个进程的环境变量,可以通过以下命令查看(文件中的内容用 \0 分割,tr 命令将 \0 转化为换行,这样看得清楚):

cat /proc/$pid/environ  | tr '\0' '\n'


Bash中新创建进程时,默认会将自己的环境变量、以及export的变量都设置为子进程的环境变量。因此,如果在一个Bash中执行:

export FOO=bar

那么这个Bash后续启动的新进程就都有FOO这个环境变量了。

Bash如果以交互形式启动,或者以login shell的方式启动,启动时会执行/etc/profile中的命令。如果/etc/profile中export了一些变量,那么这个Bash后续创建的新进程就都有这些环境变量了。

回到题主的问题。首先,“永久配置环境”,这句话本身是一个错误的说法。因为:
1、你无法修改所有已经在运行的程序的环境变量。
2、Linux下没有常规的方法可以使得所有新创建的进程都有一个预配置的环境变量。通常情况下,一个进程的环境变量由其父进程设置。如果父进程创建子进程时没有特别处理,一般子进程会直接继承父进程的环境变量。因此,如果要新创建的进程有某个特定的环境变量,那么这个变量一定是被父进程或者某个祖先进程设置的。
3、/etc/profile 只有Bash以及一些其他shell会读取,所以如果一个进程(以及其所有祖先进程)不是被bash启动的话,那么/etc/profile对他没有任何影响。(当然也有一些特例,有些非shell程序也会读取/etc/profile。)

下面解释一下,当你向/etc/profile中写入了export PATH和OMP_NUM_THREADS之后会有哪些效果:

  • 当你启动一个新的Bash进程时,这个Bash进程就会有这两个环境变量,并且它后续启动的进程也会有这两个环境变量。
  • 当你重启系统后,新开Bash进程,同上。
  • 重启系统后,由systemd启动的进程没有这两个环境变量。如果是其他init系统,则看情况了,有些init系统会读取/etc/profile,有些不会。
  • 当前python进程的环境变量没有变化,当前python进程后续启动的子进程也没有这两个环境变量,当前python进程的父进程(以及所有祖先进程)的环境变量也没有变化。


以上是对标题 “用python去修改linux环境变量为何无效?” 的回答。实际上,这个问题也同样是一个错误,或者说不完整的问题。因此,请不要追问 “如何才能生效” 的问题。如果题主理解了前面的回答,可以考虑一下:你期望哪些进程有PATH和OMP_NUM_THREADS这个环境变量?这些进程分别是怎样启动的(把所有祖先进程列出来)?如果某个进程没有这个环境变量,我们在针对具体情况再去讨论如何解决这个具体的问题。

环境变量是进程环境的属性之一,当你用os.system去执行的时候,实际上是:

1 fork一个子进程

2 子进程exec另一个shell程序,执行你要执行的命令,比如你source,实际上是exec一个shell进程再source,所以子进程的环境变量被你改了

但是你这个python进程不受任何影响

要修改当前进程的环境变量,用os.putenv,或操作os.environ这个字典(实际上并不是dict的直接实例,而是一个用法和dict基本一样的对象),这俩的区别参考os模块的文档