Linux系统 环境变量

环境变量

  • 写在前面
  • 概念
  • 查看环境变量
    • main函数的参数
      • argc & argv
      • env
  • bash环境变量

写在前面

对于环境变量,本篇主要介绍基本概念及三四个环境变量 —— PATH、HOME、PWD。其中 PATH 作为 “ 敲门砖 ”,我们会更详细讲解;理解环境变量的全局属性 —— 环境变量是可以被子进程继承(注意区分 C++ 里的继承);环境变量的组织方式。其次会介绍命令行参数 —— main 函数的参数。

概念

环境变量一般是指在操作系统中用来指定操作系统运行环境的一些参数

我们在编写 C/C++ 代码的,在链接的时候,从来不知道我们的所链接的动态静态库在哪里,但是照样可以链接成功,生成可执行程序,原因就是有相关环境变量帮助编译器进行查找。

环境变量通常具有某些特殊用途,还有在系统当中通常具有全局特性。

LInux操作系统本身就是一个用C语言写的程序,操作系统可以在运行过程中开辟空间。环境变量的本质,就是在操作系统运行过程中,为自己开辟的空间,存储了一些重要的信息。

为了更好的理解环境变量的作用,现在我们先来思考一个问题,我们在执行指令的过程之中,可以直接输入指令的名字,但是我们自己写的可执行程序,一定要加上绝对路径或者相对路径才可以运行,这是为什么呢?

其实这个过程之中,环境变量的作用就显现出来了。

PATH:一个环境变量,存储着多个路径,在这些路径下面的可执行程序,可以直接执行。

PATH是我们讲解的第一个重要的环境变量,我们现在尝试观察这个环境变量。

echo $xxx:查看xxx环境变量的内容

在这里插入图片描述
可以看出这个环境变量是由多个路径组成的,每个路径由分开。
因此我们只需要把自己的可执行程序放在这里面某一个路径下面,我们自己写的可执行程序就也可以当作指令来执行了。而这个工作就是把自己写的软件安装到系统之中。

我们也可以想办法把自己的路径加到PATH环境变量中。

xxx=:修改环境变量的值

如果我们当前路径下有一个可执行程序test.exe并且我们将其路径加到了PATH中,那么我们就可以直接执行这个程序,但是原来PATH中的内容会被我们的修改直接覆盖。导致原有的ls,pwd,mkdir等等指令全部执行不了了。但是不用惊慌,我们只需要关闭xshell,然后重启,此时PATH就会恢复原先的值。

我们再来看看几个常用的的环境变量。

USER:记录当前的用户
PWD:记录当前路径
HOME:记录家目录

在这里插入图片描述

我们还可以自己定义一些环境变量:

export xxx=:定义xxx环境变量

在这里插入图片描述
不过我们自己定义的环境变量在重新启动的时候也会失效。

还有一个与环境变量相关的重要指令。

env:输出所有环境变量。

展示部分如下。
在这里插入图片描述

查看环境变量

通过之前的认识,我们知道可以通过env来查看所有的环境变量,也可以通过echo $xxx查看单个环境变量,但是这些都是在命令行中操作的,如果我们想在可执行程序中查看需要如何做呢?

  • getenv

getenv是一个函数,其定义在<stdlib.h>中。功能是:输入一个字符串作为参数,该函数输出该字符串对应的环境变量的内容。

当前test.exe程序内容如下:

#include <stdio.h>      
#include <stdlib.h>      int main()      
{      const char* path = getenv("PATH");      const char* home = getenv("HOME");      printf("PATH = %s\n", path);      printf("HOME = %s\n", home);    return 0;    
}      

在这里插入图片描述

main函数的参数

也许你或许听说过,main函数也是有参数的,但是在学习C/C++的过程中,这个参数好像可写可不写,学习环境后,我们就可以了解一下这些变量的意义是什么了。

argc & argv

main的前两参数分别是argc和argv,传参形式如下:

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

可见,argcint类型的变量,而argv是一个char*的数组。还记得我们刚刚的env指令内容吗?其实它们两个的结构是一样的。argv每个元素都是char*类型,分别指向一个字符串,argv的最后一个元素也是NULL,用于标识argv的数组结尾。而argc代表了argv中元素的个数,所以我们既可以通过NULL来判断argv结尾,也可以通过argc来判断结尾。在test中执行如下代码,来看看argv中存储了什么:

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

在这里插入图片描述
现在argv中只有一个元素,即字符串"./test.exe"

给./test.exe加几个选项试试,./test.exe -a -b -c

在这里插入图片描述
可以得出argv参数内部,存储的是调用可执行程序时,输入的选项
通过argv存储的内容我们就可以根据不同选项执行不同的内容。

#include <stdio.h>
#include <string.h>
#include <stdbool.h>int main(int argc, char* argv[])    
{              bool flagA = false, flagB = false, flagC = false;    for(int i = 0 ; i < argc; i++)    {          if (strcmp(argv[i], "-a") == 0)    flagA = true;//说明输入了-a选项    else if (strcmp(argv[i], "-b") == 0)    flagB = true;//说明输入了-b选项    else if (strcmp(argv[i], "-c") == 0)    flagC = true;//说明输入了-c选项    }          printf("正在执行 test.exe\n");    if(flagA)    printf("功能a执行中...\n");    if(flagB)    printf("功能b执行中...\n");    if(flagC)    printf("功能c执行中...\n");    return 0;                                                                                               
}   

我们使用的绝大多数指令都是有很多选项的,我们通过输入不同的选项,让程序执行不同的功能,而程序就是通过识别argv,来判断用户输入了哪些选项,进而执行特定的功能的。

在我们向bash输入一大段指令的时候,指令本质就是一个字符串,bash会把字符串拆解为一个个小小的字符串,然后把他们整合到一个叫做命令行参数表的东西中,命令行参数表其实就是一个指针数组,而argv参数可以接收bash维护的数组,在程序内部使用。


env

main函数的第三个参数叫做env,其实它也是一个char*类型的指针数组,类型为char**

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

这个参数和指令env的内容完全一致的。存储了所有环境变量,并且以NULL结尾。

#include <stdio.h>    int main(int argc, char* argv[], char* env[])
{for(int i = 0; env[i] != NULL; i++){printf("env[%d]: %s\n", i, env[i]);}return 0;
}

bash环境变量

其实环境变量是可以继承的,可以通过下面的代码证明。

#include <stdio.h>                                                                                        
#include <unistd.h>
#include <sys/types.h>int main(int argc, char* argv[], char* env[])    
{    pid_t id = fork();    if(id == 0)//子进程    {    for(int i = 0; i < 3; i++)    {    printf("child:  env[%d]: %s\n", i, env[i]);                                                           }    }    else//父进程    {    for(int i = 0; i < 3; i++)    {    printf("father: env[%d]: %s\n", i, env[i]);    }    }    return 0;    
}  

通过fork创建了一个子进程,然后父子进程分别输出env这个数组中的前三个字符串。

输出结果:
在这里插入图片描述

可以看到父子进程都可以正常使用env,说明子进程是可以继承父进程的环境变量表的。而在所有命令行调用的进程中,都是bash的子进程,因此我们的在命令行调用的进程可以继承bash的环境变量表。只有我们登录了系统的时候,bash才会被创建,之前我们就知道,bash要维护一张环境变量表,那么bash的环境变量是怎么来的?其实这些环境变量是存储的在磁盘上的,而我们启动bash的时候,会把这些环境变量从磁盘上拷贝到内存中,组成一张环境变量表,我们访问环境变量实际上是在访问内存中的环境变量,我们修改环境变量实际上也是在修改内存中的环境变量,不会影响磁盘中的环境变量。

因此当我们重新启动Xhell时,bash都会重新从磁盘中拷贝一次环境变量,这样我们之前所做的修改或者自己添加的环境变量都会被覆盖重置

另:在家目录中,有一个叫做.bash_profile的隐藏文件 ,其内部存储的就是环境变量。这个就是我们每次重新启动从磁盘拷贝环境变量的文件内容。

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

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

相关文章

107,【7】buuctf web [CISCN2019 华北赛区 Day2 Web1]Hack World

这次先不进入靶场 看到红框里面的话就想先看看uuid是啥 定义与概念 UUID 是 Universally Unique Identifier 的缩写&#xff0c;即通用唯一识别码。它是一种由数字和字母组成的 128 位标识符&#xff0c;在理论上可以保证在全球范围内的唯一性。UUID 的设计目的是让分布式系…

cesium传感器和大量线问题记录

1&#xff1a;场景图初始加载大量轨道线。 // 创建几何体实例const geometry new Cesium.PolylineGeometry({positions: positions,width: 1.0, // 轨道线的宽度});const geometryInstance new Cesium.GeometryInstance({id: res.id,geometry: geometry,});// 创建材质实例co…

【LeetCode】5. 贪心算法:买卖股票时机

太久没更了&#xff0c;抽空学习下。 看一道简单题。 class Solution:def maxProfit(self, prices: List[int]) -> int:cost -1profit 0for i in prices:if cost -1:cost icontinueprofit_ i - costif profit_ > profit:profit profit_if cost > i:cost iret…

01vue3实战-----前言

01vue3实战-----前言 1.大前端时代2.技术栈3.项目大致展示4.创建Vue项目4.1Vue CLI4.2create-vue 5.参考资料 1.大前端时代 前端移动端iOS/android开发桌面端 window/mac 常用的electron框架来开发其它平台:穿戴设备、车载系统(智能汽车)、VR、AR…web3方向 2.技术栈 开发工…

使用request库实现接口测试-笔记

目录 request库request库的安装和查验request发送请求的语法获取响应结果指定内容案例 Unittest框架集成Requests库Unittest框架语法Unittest框架基础代码案例 request库 request库是python编写的&#xff0c;基于urllib的HTTP库,使用方便。 request库的安装和查验 安装&…

快速傅里叶离散变换FFT (更新中)

声明&#xff1a;参考了 y y c yyc yyc 的 blog 和 PPT (from smwc) &#xff0c;以及 w z r wzr wzr 的 blog 。 目录 Part 1 多项式Part 2 FFT概论Part 3 点值与插值Part 4 复数&#xff0c;单位根Part 5 Part 1 多项式 定义&#xff1a;对于有限数列 A 0 A_{0} A0​~ n…

【C++】STL——list的使用

目录 &#x1f495;1.带头双向链表List &#x1f495;2.list用法介绍 &#x1f495;3.list的初始化 &#x1f495;4.size函数与resize函数 &#x1f495;5.empty函数 &#x1f495;6.front函数与back函数 &#x1f495;7.push_front,push_back,pop_front,pop_back函数…

2021Java面试-基础篇

文章目录 前言一&#xff1a; Java概述 1、何为编程2、JDK1.5之后的三大版本3、JVM,JRE和JDK的关系4、什么是跨平台&#xff1f;原理是什么5、Java语言有哪些特点6、什么是字节码&#xff1f;采用字节码的最大好处是什么7、什么是Java程序的主类&#xff1f;应用程序和小程序的…

MapStruct工具类的使用

文章目录 1.简介2.作用2.1 属性拷贝2.2 类型安全2.3 性能高效2.4 自定义转换 3.如何使用3.1 创建一个maven工程项目并引入mapstruct的坐标依赖&#xff1a;3.2 源对象类3.3 目标对象类&#xff08;和源对象类中的字段属性名相同&#xff09;3.4 创建接口SyncMapper&#xff08;…

FPGA| 使用Quartus II报错Top-level design entity ““ is undefined

1、使用FPGA准备点亮LED测试下板子&#xff0c;发现这个报错Error (12007): Top-level design entity "LEDLED" is undefined 工程如上图 报错如下图 2、分析到原因是因为工程名称和顶层模块里面的module名称不一样导致 解决办法&#xff1a;修改module名称和顶层模…

Window获取界面空闲时间

‌GetLastInputInfo‌是一种Windows API函数&#xff0c;用于获取上次输入操作的时间。 该函数通过LASTINPUTINFO结构返回最后一次输入事件的时间。 原型如下 BOOL WINAPI GetLastInputInfo(PLASTINPUTINFO plii);那么可以利用GetLastInputInfo来得到界面没有操作的时长 uint…

LeetCode - #198 打家劫舍

网罗开发 &#xff08;小红书、快手、视频号同名&#xff09; 大家好&#xff0c;我是 展菲&#xff0c;目前在上市企业从事人工智能项目研发管理工作&#xff0c;平时热衷于分享各种编程领域的软硬技能知识以及前沿技术&#xff0c;包括iOS、前端、Harmony OS、Java、Python等…

验证工具:SVN版本控制

1-SVN概念 SVN(Subversion)是一种集中式版本控制系统,它用于文件和目录的版本管理,允许多个用户协同工作,同时追踪每个文件和目录的历史修改记录。以下是关于SVN版本控制的详细介绍: 一、SVN的基本概念 仓库(Repository):SVN的仓库是一个集中存储所有文件和目录的地…

【DeepSeek】本地私有化部署 DeepSeek 模型教程

一、引言 DeepSeek 模型是一种强大的语言模型&#xff0c;本地私有化部署可以让用户在自己的环境中安全、高效地使用该模型&#xff0c;避免数据传输到外部带来的安全风险&#xff0c;同时也能根据自身需求进行定制化配置。本教程将详细介绍如何在本地进行 DeepSeek 模型的私有…

【玩转 Postman 接口测试与开发2_016】第13章:在 Postman 中实现契约测试(Contract Testing)与 API 接口验证(上)

《API Testing and Development with Postman》最新第二版封面 文章目录 第十三章 契约测试与 API 接口验证1 契约测试的概念2 契约测试的工作原理3 契约测试的分类4 DeepSeek 给出的契约测试相关背景5 契约测试在 Postman 中的创建方法6 API 实例的基本用法7 API 实例的类型实…

为什么“记住密码”适合持久化?

✅ 特性 1&#xff1a;应用重启后仍需生效 记住密码的本质是长期存储用户的登录凭证&#xff08;如用户名、密码、JWT Token&#xff09;&#xff0c;即使用户关闭应用、重启设备&#xff0c;仍然可以自动登录。持久化存储方案&#xff1a; React Native 推荐使用 AsyncStorag…

echarts、canvas这种渲染耗时的工作能不能放在webworker中做?

可以将 ECharts、Canvas 等渲染耗时的工作放在 Web Worker 中进行处理。Web Worker 允许在后台线程中运行 JavaScript&#xff0c;从而将计算密集型任务从主线程中分离出来&#xff0c;避免阻塞用户界面。以下是一些关键点&#xff1a; 优势 性能提升&#xff1a;将耗时的渲染…

proxmox通过更多的方式创建虚拟机

概述 作为一名资深运维工程师&#xff0c;我们经常需要在 Proxmox 虚拟化平台上创建和管理虚拟机。本文将介绍三种不同的方式在 Proxmox 上创建 Ubuntu 虚拟机&#xff1a; 通过 Proxmox 命令创建虚拟机通过 Shell 脚本自动化创建虚拟机使用 Proxmox API 创建虚拟机 每种方式…

【分布式架构理论3】分布式调用(2):API 网关分析

文章目录 一、API 网关的作用1. 业务层面&#xff1a;简化调用复杂性2. 系统层面&#xff1a;屏蔽客户端调用差异3. 其他方面&#xff1a; 二、API 网关的技术原理1. 协议转换2. 链式处理3. 异步请求机制1. Zuul1&#xff1a;同步阻塞处理2. Zuul2&#xff1a;异步非阻塞处理 三…

使用线性回归模型逼近目标模型 | PyTorch 深度学习实战

前一篇文章&#xff0c;计算图 Compute Graph 和自动求导 Autograd | PyTorch 深度学习实战 本系列文章 GitHub Repo: https://github.com/hailiang-wang/pytorch-get-started 使用线性回归模型逼近目标模型 什么是回归什么是线性回归使用 PyTorch 实现线性回归模型代码执行结…