『 Linux 』环境变量

news2025/1/12 6:20:53

文章目录

    • 🚀什么是环境变量🚀
    • 🚀查看环境变量🚀
      • 🕹️和环境变量有关的命令🕹️
    • 🚀PATH环境变量🚀
      • 🕹️设置PATH环境变量🕹️
    • 🚀HOME环境变量🚀
    • 🚀通过代码获取环境变量🚀
      • 🕹️main函数的形参🕹️
        • 🎮 命令行参数
        • 🎮 利用main函数的第三个参数 -- char * env[]获取环境变量
        • 🎮 环境变量全局性的证明
      • 🕹️利用全局变量 char **environ获取环境变量🕹️
      • 🕹️利用getenv()函数获取环境变量🕹️
    • 🚀Linux中的环境变量与全局变量🚀
      • 🕹️使用set显示变量🕹️


请添加图片描述

🚀什么是环境变量🚀

在c/C++当中,根据不同程序的需要,我们需要为该程序定义某些变量int _data,double _price;

这些变量本质上就是一段内存空间,用来存储程序需要的值或数据;

而在OS中,OS本身在运行当中也需要一些必要的值或者数据,所以在OS当中也会存在一些变量用来存储这些所谓必要的值或者数据参数;

本质上来说环境变量(environment variables)一般是指操作系统中指定操作系统运行环境的一些参数;

一般常见的环境变量有:

  • PATH
    指定命令的搜索路径;
  • HOME
    指定用户的主工作目录(即用户登陆到Linux系统当中时的默认路径);
  • SHELL
    当前Shell,它的值通常为/bin/bash;
    不同的环境变量有着不同的功能;

请添加图片描述

🚀查看环境变量🚀

在Linux当中可以使用env命令打印所有的环境变量;

查看环境变量一般的方式是利用echo命令将环境变量进行打印;
语法:

echo $NAME
  • 其中:

    NAME指环境变量名

    echo命令在Linux中的作用为:用于显示消息或输出其他命令的结果;

    其中若是使用echo打印环境变量时必须使用$符,否则将会直接将环境变量名当作字符串进行打印;

    $ echo PATH
    PATH
    $ echo $PATH
    /usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/_user/.local/bin:/home/_user/bin
    

请添加图片描述

🕹️和环境变量有关的命令🕹️

  • echo:显示某个环境变量值;

  • export:设置一个新的环境变量;

  • env:显示所有环境变量;

  • unset:清除环境变量;

  • set:显示本地定义的shell变量和环境变量;


请添加图片描述

🚀PATH环境变量🚀

存在一个程序mytest:

其主要的作用为打印hello world;

std::cout << "hello world"<< std::endl ;

$ /usr/bin/ls
Makefile  mytest  test.cpp
$ ls
Makefile  mytest  test.cpp
$ ./mytest 
hello world
$ mytest
-bash: mytest: command not found

已知ls本身就是一个程序,且该程序存在于路径usr/bin/ls;

在该程序中运行了两个程序:mytestls,分别为用户(我)创建的程序mytest与OS中自带的命令程序ls;

但是在使这两个程序运行时的方式并不同,在运行用户的程序时必须使用相对路径或者绝对路径从而运行程序,而在使用OS自身的程序时对路径的需求却可有可无;

很显然这个原因是因为:

本质上的原因是因为在OS当中,无论是运行什么程序都需要保证首先得找到这个程序(即拥有这个程序的路径)才能将该程序进行运行;
而例如ls之类的这样的命令,OS默认是可以直接找到它的,因为这些命令的所在位置都位于环境变量PATH当中;

  • 使用 echo 打印环境变量$PATH:

    $ echo $PATH
    /usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/_user/.local/bin:/home/_user/bin
    

    从打印出来的结果中可以发现,PATH环境变量以/.../..作为路径,:作为分隔符;

  • 使用which找到ls命令所在路径:

    which命令将在PATH环境变量中寻找某个指令的地址;

    $ which ls
    alias ls='ls --color=auto'
      	/usr/bin/ls
    

    ls的所在路径正是在PATH环境变量当中;

    当在Linux命令行上输入对应的命令时,OS将会在PATH环境变量中寻找该命令的地址;
    顺序为从左至右且以:作分隔符;

    /usr/local/bin,/usr/bin,/usr/local/sbin


请添加图片描述

🕹️设置PATH环境变量🕹️

若是想使自己的程序设置成类似于ls命令;

可以使用两种方式:

  • 将文件移至PATH环境变量的任意路径当中:

    由于PATH环境变量中所存储的命令或者文件都是以路径的形式进行存储;

    所以可以将需要的文件添加至PATH环境变量中的任意路径;

    但是实际上这种方式一般是不可取的:

    虽然以这种方式可以达到预期的效果:使得自己的程序能像命令一样直接运行
    但是对应的有很大的副作用;
    由于PATH环境变量内的文件是OS预先设置好的,由于这种操作在Linux当中相当于将自己写的命令安装至OS中;
    所以如果贸然的将自己的程序添加至路径当中可能会出现污染命令池;

  • 使用export命令将程序加入PATH环境变量:

    语法:

    export PATH=$PATH:程序路径
    

    使用pwd查看可执行程序所在路径以及使用echo命令打印当前PATH环境变量:

    $ pwd
    /home/_user/Begin/my_-linux/Pro23/Environment1201/T1201
    $ echo $PATH
    /usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/_user/.local/bin:/home/_user/bin
    

    使用export将路径加入到PATH环境变量后再对PATH环境变量进行打印:

    export PATH=$PATH:/home/_user/Begin/my_-linux/Pro23/Environment1201/T1201  # 设置
    $ echo $PATH  # 打印
    /usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/_user/.local/bin:/home/_user/bin:/home/_user/Begin/my_-linux/Pro23/Environment1201/T1201
    

    此时多了一个/home/_user/Begin/my_-linux/Pro23/Environment1201/T1201的路径;

    不带路径直接运行该程序:

    $ mytest
    hello world
    

    同时使用which也可以在$PATH环境变量中找到该程序:

    $ which mytest
    ~/Begin/my_-linux/Pro23/Environment1201/T1201/mytest
    

    同时这种方式在测试设置$PATH环境变量较为安全;

    因为在不改配置文件的前提下只使用命令行的模式(类似于export)的方式对环境变量进行修改时,修改效果只在当前对话当中;

    当重新登陆该会话时环境变量将恢复:

    $ echo $PATH
    /usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/_user/.local/bin:/home/_user/bin
    

请添加图片描述

🚀HOME环境变量🚀

HOME环境变量即为指定用户的主工作目录(即用户登陆到Linux系统当中时的默认路径);

实际上不同的用户使用echoHOME环境变量进行打印时所得到的结果是不同的:

  #------_user用户下------ 
$ echo $HOME
/home/_user

  #------root用户下------ 
# echo $HOME
/root

请添加图片描述

🚀通过代码获取环境变量🚀

除了以命令行的形式echo $变量名,env获取环境变量以外也可以通过程序,即以代码的形式获取环境变量;


请添加图片描述

🕹️main函数的形参🕹️

实际上在main函数当中也有三个形参,分别为:

  • int argc

    向main函数传递的参数个数;

  • char *argv[]

    代表执行的程序名称和执行程序时输入的参数;

  • char *env[]
    环境变量信息;

int main(int argc , char *argv[] , char *env[])
{;}

其中int argcchar *argv[]都为命令行参数;


请添加图片描述

🎮 命令行参数

main()函数的参数当中,其中int argcchar *argv[]均为命令行参数;

可以通过这两个参数实现程序在不同的指令下做到不同的运行结果;

存在一个程序(mytest):

#include<iostream> 
using namespace std;

int main(int argc , char *argv[] , char *env[])
{
    for(int i = 0;i<argc;++i){
    printf("argv[%d]:%s\n",i,argv[i]);
  }
  return 0;
}

运行结果:

$ ./mytest -a -b -c -d -e -f
argv[0]:./mytest
argv[1]:-a
argv[2]:-b
argv[3]:-c
argv[4]:-d
argv[5]:-e
argv[6]:-f

可根据该运行结果配合Linux中类似于ls -a等命令进行联想;


请添加图片描述

🎮 利用main函数的第三个参数 – char * env[]获取环境变量

该参数的类型为一个指针数组类型,一个数组内所存储的数据都为char*的指针;

而这个形参即为启动该进程的进程(即这个进程的父进程)传给该进程的环境变量信息;

可以推断出这个指针数组内的char*指针实际上为字符串,而启动该进程的进程将环境变量信息以字符串的形式传递给该进程;
在这里插入图片描述

可以使用一个程序进行证明 (以字符串的形式打印环境变量);

存在一个程序(mytest):

#include<iostream> 
using namespace std;
int main(int argc , char *argv[] , char *env[])
{
	for(int i = 0;env[i];++i){
  		cout<<env[i]<<endl;
	}
	return 0;
}
/*
* env[]为指针数组;
* 遍历该数组并以字符串的形式打印出每个char*字符串;
*/

在Linux下环境变量一般是以字符串的形式利用指针数组进行存储,并以指针数组的形式将其传递给main函数;


请添加图片描述

🎮 环境变量全局性的证明

实际上在一个程序当中,main()函数当中的char*env[]中的环境变量信息来自于其父进程;

即也验证了这个形参即为启动该进程的进程(即这个进程的父进程)传给该进程的环境变量信息;

验证:

  • 存在一个程序(mytest):

    #include<iostream> 
    using namespace std;
    int main(int argc , char *argv[] , char *env[])
    {
    	for(int i = 0;env[i];++i){
    		cout<<env[i]<<endl;
    	}
    	return 0;
    }
    

    可以使用该程序获取当前的所有环境变量;

    通过该程序配合| grep可以查看该程序所接收的众多环境变量中的部分环境变量;

  • 设置临时环境变量:

    在该对话中并不存在一个MYTEST的环境变量;

    所以如果运行该程序并配合| grep MYTEST也将不会有对应的结果;

    $ ./mytest | grep MYTEST
    $ 
    

    使用export命令设置一个临时的环境变量,且该环境变量的值为hello_world;

    $ export MYTEST=hello_world
    
  • 再次运行程序同时再次使用| grep MYTEST搜索该环境变量:

    当再次搜索该环境变量时则可以找到;

    $ ./mytest | grep MYTEST
    MYTEST=hello_world
    

由于根据该程序中所打印的环境变量都是由main()函数的第三个参数char *env[]进行打印的;

由此可以推断出该程序的环境变量是由其父进程所传递,对应的来说一个进程①在运行另一个进程②时,进程①会将自身的环境变量传给进程②;

同时也可以得出一个结论:

环境变量将从配置文件开始以多叉树的形式分发给每一个进程(父子进程间环境变量的传递),因此可以认为环境变量是具有全局属性的;


请添加图片描述

🕹️利用全局变量 char **environ获取环境变量🕹️

同时在Linux中除了该种方式可以获取环境变量以外,还拥有一个全局的环境变量char** environ,且其对于获取环境变量所调用的方式与使用main()函数的第三个参数char *env[]所调用的方式相同;

示例:

#include<iostream> 
using namespace std;
int main(int argc , char *argv[] , char *env[]){
	extern char **environ; 
	for(int i = 0;environ[i];++i){
		cout<<environ[i]<<endl;
	}
}

由于environ并不包含与任何头文件,所以在使用该全局变量前必须使用extern进行声明;


请添加图片描述

🕹️利用getenv()函数获取环境变量🕹️

getenv()函数是包含于<stdlib.h>头文件中的一个函数;

该函数能够通过所给的环境变量(字符串)返回对应的环境变量数据;

其语法为:

char *getenv(const char *name);

示例:

#include<iostream> 
using namespace std;
int main(int argc , char *argv[] , char *env[])
{
  cout<<getenv("PATH")<<endl;
  return 0;
}

调用:

$ ./mytest 
/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/_user/.local/bin:/home/_user/bin

通常来说大部分情况使用该方式获取对应的环境变量的所使用的频率要高于上面的两种;

其中最主要的原因是若是使用上面的两种方式获取环境变量则将会有大量的环境变量以字符串的形式进行展现,需要使用者进行二次甄别;


请添加图片描述

🚀Linux中的环境变量与全局变量🚀

在上文中所提到的都为环境变量;

而在Linux中不仅可以定义环境变量还可以定义全局变量;

  • 环境变量的定义:

    export var_name=_value
    

    示例:

    $ export Test1=hello_world # 设置环境变量
    $ echo $Test1 # 使用echo打印变量
    hello_world
    $ env | grep Test1 # 可以使用env显示所有环境变量,配合grep显示出对应的环境变量
    Test1=hello_world
    
  • 局部变量的定义:

    var_name=_value
    

    示例:

    $ Test2=HELLO_WORLD # 设置环境变量
    $ echo $Test2 # 使用echo打印变量
    HELLO_WORLD
    $ env | grep Test2 # 可以使用env显示所有环境变量,配合grep显示出对应的环境变量
    $ # 无结果显示
    

请添加图片描述

🕹️使用set显示变量🕹️

在Linux当中可以使用set显示所有的变量 (包括局部变量与环境变量);

$ set | grep Test1
Test1=hello_world
$ set | grep Test2
Test2=HELLO_WORLD

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1276252.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

手敲单链表,简单了解其运行逻辑

1. 链表 1.1 结构组成 链表是一种物理存储结构上非连续存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的引用链接次序实现的 。 链表的结构如下图所示&#xff0c;是由很多个节点相互通过引用来连接而成的&#xff1b;每一个节点由两部分组成&#xff0c;分别数据域&…

【LeetCode每日一题合集】2023.11.20-2023.11.26 (二叉树中的伪回文路径)

文章目录 53. 最大子数组和解法1——DP解法2——分治&#xff08;维护区间、类似线段树的思想&#xff09; 2216. 美化数组的最少删除数&#xff08;贪心&#xff09;2304. 网格中的最小路径代价1410. HTML 实体解析器&#xff08;模拟&#xff09;2824. 统计和小于目标的下标对…

iOS Class Guard 成功了,但无法区分差异

​ 我正在开发一个静态库&#xff0c;并使用 Polidea 的 iOS Class Guard 来混淆我的静态库。我按照步骤在项目的根路径中下载 obfuscate_project&#xff0c;更改其中所需的名称&#xff0c;最后在终端中运行 bash obfuscate_project。我收到一条消息&#xff0c;说我的构建成…

【漏洞复现】大华智慧园区综合管理平台deleteFtp接口远程命令执行

漏洞描述 大华智慧园区综合管理平台deleteFtp接口存在远程命令执行,攻击者可利用该漏洞执行任意命令,获取服务器控制权限。 免责声明 技术文章仅供参考,任何个人和组织使用网络应当遵守宪法法律,遵守公共秩序,尊重社会公德,不得利用网络从事危害国家安全、荣誉和利益…

CMMI认证含金量高吗

一、CMMI认证含金量解答 CMMI&#xff0c;即能力成熟度模型集成&#xff0c;是由美国卡内基梅隆大学软件工程研究所开发的一种评估企业软件开发过程成熟度的模型。CMMI认证的含金量究竟高不高呢&#xff1f;答案是肯定的。CMMI认证被誉为软件开发行业的“金牌标准”&#xff0…

spring cloud gateway源码分析,一个请求进来的默认处理流程

1.前言 spring cloud gateway的基本组成和作用就不细赘述&#xff0c;此篇适合对此有一定了解的人阅读。 spring cloud gateway版本: Hoxton.SR1 spring cloud gateway的配置使用yml配置&#xff1a; server:port: 9527y#根据微服务名称进行动态路由的配置 spring:applicati…

zookeeper心跳检测 (实操课程)

本系列是zookeeper相关的实操课程&#xff0c;课程测试环环相扣&#xff0c;请按照顺序阅读来学习和测试zookeeper。 阅读本文之前&#xff0c;请先阅读----​​​​​​zookeeper 单机伪集群搭建简单记录&#xff08;实操课程系列&#xff09;zookeeper 客户端常用命令简单记录…

人工智能-优化算法之学习率调度器

学习率调度器 到目前为止&#xff0c;我们主要关注如何更新权重向量的优化算法&#xff0c;而不是它们的更新速率。 然而&#xff0c;调整学习率通常与实际算法同样重要&#xff0c;有如下几方面需要考虑&#xff1a; 首先&#xff0c;学习率的大小很重要。如果它太大&#xf…

知识管理平台Confluence:win10安装confluence

文章目录 介绍主要功能 安装教程安装java运行平台JRE安装数据库Postgresql在Postgresql创建confluence使用的数据库创建数据库用户创建数据库 安装confluence注册confluence启动confluence 参考链接 介绍 Confluence 是由澳大利亚软件公司 Atlassian 开发的企业协作平台。它提…

flutter开发实战-ValueListenableBuilder实现局部刷新功能

flutter开发实战-ValueListenableBuilder实现局部刷新功能 在创建的新工程中&#xff0c;点击按钮更新counter后&#xff0c;通过setState可以出发本类的build方法进行更新。当我们只需要更新一小部分控件的时候&#xff0c;通过setState就不太合适了&#xff0c;这就需要进行…

canvas基础:渲染文本

canvas实例应用100 专栏提供canvas的基础知识&#xff0c;高级动画&#xff0c;相关应用扩展等信息。 canvas作为html的一部分&#xff0c;是图像图标地图可视化的一个重要的基础&#xff0c;学好了canvas&#xff0c;在其他的一些应用上将会起到非常重要的帮助。 文章目录 示例…

java设计模式学习之【桥接模式】

文章目录 引言桥接模式简介定义与用途&#xff1a;实现方式 使用场景优势与劣势桥接模式在Spring中的应用绘图示例代码地址 引言 想象你正在开发一个图形界面应用程序&#xff0c;需要支持多种不同的窗口操作系统。如果每个系统都需要写一套代码&#xff0c;那将是多么繁琐&am…

scrapy爬虫中间件和下载中间件的使用

一、关于中间件 之前文章说过&#xff0c;scrapy有两种中间件&#xff1a;爬虫中间件和下载中间件&#xff0c;他们的作用时间和位置都不一样&#xff0c;具体区别如下&#xff1a; 爬虫中间件&#xff08;Spider Middleware&#xff09; 作用&#xff1a; 爬虫中间件主要负…

SQL Server 2016(基本概念和命令)

1、文件类型。 【1】主数据文件&#xff1a;数据库的启动信息。扩展名为".mdf"。 【2】次要&#xff08;辅助&#xff09;数据文件&#xff1a;主数据之外的数据都是次要数据文件。扩展名为".ndf"。 【3】事务日志文件&#xff1a;包含恢复数据库的所有事务…

深入理解前端路由:构建现代 Web 应用的基石(下)

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

2024年天津天狮学院专升本专业课报名缴费流程

天津天狮学院高职升本缴费流程 一、登录缴费系统 二、填写个人信息&#xff0c;进行缴费 1.在姓名处填写“姓名”&#xff0c;学号处填写“身份证号”&#xff0c;如下图所示&#xff1a; 此处填写身份证号 2.单击查询按钮&#xff0c;显示报考专业及缴费列表&#xff0c;…

JPA数据源Oracle异常记录

代码执行异常 ObjectOptimisticLockingFailureException org.springframework.orm.ObjectOptimisticLockingFailureException: Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1; nested exception is org.hibernate.StaleSta…

从0开始学习JavaScript--JavaScript ES6 模块系统

JavaScript ES6&#xff08;ECMAScript 2015&#xff09;引入了官方支持的模块系统&#xff0c;使得前端开发更加现代化和模块化。本文将深入探讨 ES6 模块系统的各个方面&#xff0c;通过丰富的示例代码详细展示其核心概念和实际应用。 ES6 模块的基本概念 1 模块的导出 ES…

java原子类型

AtomicBoolean AtomicInteger AtomicLong AtomicReference<V> StringBuilder - 不是原子类型。StringBuilder 是 java.lang 包下的类 用法&#xff1a;无需回调改变数值

基于springboot + vue框架的网上商城系统

qq&#xff08;2829419543&#xff09;获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;springboot 前端&#xff1a;采用vue技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xf…