OS21.【Linux】环境变量

目录

1.与环境变量有关的实验

A.对比命令和自制程序的运行

为什么.像ls、pwd这样的命令运行是不需要加路径?

执行自制程序而不加路径的方法,看看PATH环境变量

方法1:将自制程序移动到系统的搜索路径下

方法2:临时修改PATH环境变量

B.查看系统中所有环境变量

解释几个常见的环境变量

C.获取环境变量的函数getenv()

2.环境变量

解释全局属性

命令行参数

a.out子进程是如何得到环境变量表的?

证明能被子进程继承,也就证明的全局属性

添加环境变量的命令:export

取消环境变量的命令:unset

使用第三方变量environ来访问环境变量

3.系统变量的分类

环境变量

本地变量

set命令

证明本地变量不可被继承

4.系统命令的分类

常规命令和内建命令

man和help的区别

5.总结获取环境变量的几种方法

6.拓展阅读:自定义入口函数

7.附: C89标准文档对main函数的描述


1.与环境变量有关的实验

从与环境变量有关的实验来感受环境变量

A.对比命令和自制程序的运行

1.像ls、pwd这样的命令运行是不需要加路径的

2.如果不配置环境变量,那么运行开发者自制程序是需要加路径的

例如:当前目录下有一个a.out可执行文件,那么运行时需要加路径:

./a.out #.表示当前目录

a.out是由以下代码产生的: 

#include <stdio.h>
int main()
{printf("Hello World!");return 0;
}

为什么.像ls、pwd这样的命令运行是不需要加路径?

之前在OS4.【Linux】基本指令入门(3)文章讲过:系统的命令存储在特定的路径下,那么bash执行指令会到特定的路径中找

特定的路径存储在环境变量PATH中

 使用echo命令打印系统的环境变量:

echo $PATH #注意: $作用:说明PATH是变量名

那么在搜索命令时,会从左向右按顺序查找这些路径下的命令,如果找到了,之后的路径就不找了 

还可以打印其他环境变量:

echo $HOME #查看当前用户的家目录

 

结论:

1.PATH存储了Linux系统命令的搜索路径

2.执行非系统命令的程序需要加路径,因为PATH中没有;执行系统的命令不需要带路径,因为PATH存储了搜索路径

执行自制程序而不加路径的方法,看看PATH环境变量

方法1:将自制程序移动到系统的搜索路径下

之前OS4.【Linux】基本指令入门(3)在文章做过,这里不再演示

方法2:临时修改PATH环境变量

*注意:以下两种方法都只是临时修改PATH环境变量,重新连接服务器时PATH会恢复默认值,可以得出: PATH是内存中的环境变量

1.直接覆盖掉PATH:

 PATH=/home/guest

(按照赋值号=的规则: PATH的新值就是/home/guest) 

会导致有些系统命令是用不了的:

例如ls命令不可用,但pwd是可用的,具体为什么pwd可用本文后面会讲

这样就不用添加路径了:

2.对PATH添加新的路径

PATH=$PATH:/home/guest

(要使用冒号分割符,因为冒号分割符的作用就是分开各个路径)

这样就不用添加路径了:

B.查看系统中所有环境变量

使用不带任何选项的env命令:

其中,=前面的大写单词指的是环境变量名, =后面的是该环境变量的值

环境变量名 = 对应的值

解释几个常见的环境变量

USER

指的是当前登录的用户(随su命令切换用户而变)

LOGNAME

和USER不同的是:在某些情况下,它的值可能会保持不变,即使通过su命令切换到其他用户

PWD

指当前处于的绝对路径,其值会在每次执行cd命令后时更新

*注:pwd命令不是使用了PWD环境变量,而是通过getcwd函数实现的

HOME

指当前用户的家目录

SHELL

指当前Shell,它的值通常是/bin/bash

C.获取环境变量的函数getenv()

看看手册:

getenv()的作用:获取指定的环境变量(environment variable),需要包含<stdlib.h>头文件

函数的声明: char* getenv(const char *name);

注意到有两个参数:

示例代码:

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

运行结果:和不带任何选项的env命令打印的结果是一样的

2.环境变量

从上面的三个实验可以总结出环境变量的定义:

环境变量: 是操作系统中用来指定操作系统运行环境的一些参数,是系统提供的一组 name = value 形式的变量,不同的环境变量有不同的用途,通常具有全局属性

例如编写C/C++代码的时候,在链接的时候,有相关环境变量帮助编译器进行查找所链接的动态静态库

解释全局属性

命令行参数

回顾分析命令行参数的文章:

110.【C语言】编写命令行程序(1)

113.【C语言】编写命令行程序(2)

114.【C语言】实战分析命令行程序:srom转换工具源码分析

一般情况下,main函数用的比较多的两种写法:

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

但Linux操作系统为main函数提供了第3个参数:char *envp[] 

(来自GNU Program-Arguments网站) 

注:编译器会根据情况处理main函数的三种不同写法

envp的全称是environment variables pointer array,是环境变量指针数组,可以打印内容

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

提示: argv和env表的结构都一样

运行结果:

结论:

1.程序运行需要两张核心向量表: char* argv[],char* envp[],指针数组的结尾元素的值为空指针 2.main函数之前还有一些函数,这些函数负责向main函数传递这两张表的指针和参数的个数argc

a.out子进程是如何得到环境变量表的?

上方运行了a.out子进程,子进程打印了操作系统的所有环境变量

bash在启动时,会从操作系统的配置文件中读取环境变量信息,bash产生的子进程会继承bash得到的环境变量!

证明能被子进程继承,也就证明的全局属性

添加环境变量的命令:export

例如添加临时环境变量:

export MY_ENV=1

注意: 不能写成MY_ENV=1,这样写是添加本地变量;变成环境变量需要加export前缀

 检查是否成功添加:

env | grep MY_ENV

再次执行a.out,会发现MY_ENV已经被bash的子进程a.out继承

取消环境变量的命令:unset

使用第三方变量environ来访问环境变量

查询environ的含义:

extern char **environ其实是存一些环境变量字符串指针的数组, 最后一个元素的值为空指针

libc中定义的全局变量environ指向环境变量表,environ没有包含在任何头文件中,所以在使用时要用extern声明

例如以下代码:

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

运行结果:

 

3.系统变量的分类

系统变量主要分为本地变量和环境变量

环境变量

见上方内容

本地变量

创建本地变量的命令

本地变量名 = 本地变量值 #前面不要加export

 本地变量和环境变量不同,本地变量主要作用范围是当前shell进程,不会被子进程继承,但环境变量可以被子进程继承

例如添加2个本地变量:

 (\是续行符,可以帮助开发者写多行命令)

环境变量中是查不到的:

set命令

作用: 查系统中所有的变量(本地变量+环境变量)

例如查之前添加的本地变量:

证明本地变量不可被继承

证明bash的子进程无法获取上面提到的VAL1和VAL2的值即可

#include <stdio.h>
#include <stdlib.h>
int main()
{printf("VAL1=%s",getenv("VAL1"));printf("VAL2=%s",getenv("VAL2"));return 0;
}

运行结果:值为空 

将VAL1和VAL2用export导出,再打印:

4.系统命令的分类

常规命令和内建命令

问题:执行echo命令让bash产生了子进程,那为什么能获取本地变量?

显然echo不是常规命令

Linux下有两种命令:

1.常规命令: 通过创建子进程;来执行常规命令
2.内建命令(也称内置命令,英文名为builtin): bash 不创建子进程,而是由自己亲自执行,类似于 bash 调用了自己写的或者系统提供的函数

例如cd命令调用了chdir()系统调用接口,是由bash自己执行的函数,cd命令的作用是让进程改变自己的路径

成功切换返回0,否则返回-1 

可以写一个简单的切换路径的程序:

#include <stdio.h>
#include <unistd.h>
int main(int argc, char* argv[])
{if (argc == 1){printf("Usage:cd path");return -1;}if (chdir(argv[1]))printf("%s: No such directory\n", argv[1]);elseprintf("Change path successfully\n");while (1){scanf("%s", argv[1]);if (chdir(argv[1]))printf("%s: No such directory\n", argv[1]);elseprintf("Change path successfully\n");}return 0;
}

注意:这个程序和cd命令还不太一样, 是让进程改变自己的路径,不是更改bash的路径,因此使用上方代码切换路径后,pwd命令打印的值是不变的

可以在cwd文件中看到: 

当然可以继续切换看看cwd的值的变化

man和help的区别

man主要查看的是外部命令和一些系统调用和函数,例如ls、printf……
help主要查看的就是内置命令了。

提示:使用type命令可以查看命令的类型,例如

5.总结获取环境变量的几种方法

1.使用命令env

2.使用getenv()函数

3.直接打印char* envp[]

4.第三方变量environ获取

6.拓展阅读:自定义入口函数

Linux下可以通过自定义入口函数的方法来达到不从main函数(cppreference main_function)开始执行的目的

在https://www.geeksforgeeks.org/c/write-running-c-code-without-main/提到一种方法;

#include<stdio.h> 
#include<stdlib.h> // entry point function 
int nomain(); void _start(){ // calling entry point nomain(); exit(0); 
} int nomain() 
{ puts("Geeksforgeeks"); return 0; 
}

编译命令:

gcc -o nomain.out nomain.c -nostartfiles

 运行结果:

摘自stackoverflow what-is-the-use-of-start-in-c的回答:

The symbol _start is the entry point of your program. That is, the address of that symbol is the address jumped to on program start. Normally, the function with the name _start is supplied by a file called crt0.o which contains the startup code for the C runtime environment. It sets up some stuff, populates the argument array argv, counts how many arguments are there, and then calls main. After main returns, exit is called.

If a program does not want to use the C runtime environment, it needs to supply its own code for _start. For instance, the reference implementation of the Go programming language does so because they need a non-standard threading model which requires some magic with the stack. It's also useful to supply your own _start when you want to write really tiny programs or programs that do unconventional things.

翻译: _start符号是Linux下规定的函数的入口点,程序启动时会跳转到该符号所表示的地址

正常情况下, 由_start标识的的函数由crt0.o文件提供,它包含了 C 运行时环境的启动代码. 该代码会完成一些初始化工作: 填充参数数组 argv ,统计参数个数argc,然后调用main函数. main 返回后,再调用exit函数

如果某个程序不想使用C运行时环境,就必须自行提供 _start 的实现
当你想编写体积极小的程序,或者做一些非常规操作时,自己写 _start 也很有用

7.附: C89标准文档对main函数的描述

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

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

相关文章

加密流量论文复现:《Detecting DNS over HTTPS based data exfiltration》(上)

本文将以我个人的理解去阅读该篇流量加密论文&#xff0c;并在下一篇尽力对其中的实验部分进行复现。话不多说&#xff0c;先从论文开始着手。 内容介绍 传统的DNS(Domain Name System)协议是以明文传输的。DNS作为互联网的基础设施&#xff0c;最初设计时主要考虑的是功能和效…

Apache RocketMQ 中Message (消息)的核心概念

好的&#xff0c;我们来深入理解一下 Apache RocketMQ 中 Message (消息) 这个核心概念。这份文档详细阐述了消息的定义、在模型中的位置、内部属性、约束和使用建议。 你可以将 Message 看作是 RocketMQ 系统中数据传输和处理的最小原子单位。它承载了业务数据&#xff0c;并附…

C 语言问题

1. C语言中 union 与 struct 的区别类型structunion内存分配机制编译器为每个成员‌独立分配内存空间&#xff0c;总内存大小 所有成员大小之和&#xff08;考虑内存对齐&#xff09;所有成员‌共享同一段内存空间&#xff0c;总内存大小 ‌最大成员的大小‌数据存储特性1. 所…

[ LeetCode优选算法专题一双指针-----盛最多的水]

1.题目链接 LeetCode盛最多的水 2.题目描述 3.题目解析 问题本质分析 "盛最多水的容器" 问题可以抽象为&#xff1a;在坐标轴上有 n 条垂直线段&#xff0c;第 i 条线段的两个端点分别是 (i, 0) 和 (i, height [i])。找到两条线段&#xff0c;使得它们与 x 轴共同…

旧笔记本电脑如何安装飞牛OS

01引言随着电子产品的更新换代&#xff0c;我们有很多的电子产品已经满足不了现在的工作需求和日常娱乐了&#xff0c;比如&#xff1a;用了很久厚重笔记本电脑放在现在办公也是有点吃力了&#xff0c;我们现在换新了旧的还不想放在那里吃灰&#xff0c;怎么办呢&#xff1f;我…

某金服Java面试终极指南:25题完整解析与场景化方案

涵盖分布式锁、缓存、事务、高并发等金融系统核心考点&#xff0c;附解决方案与抗风险设计一、分布式锁深度解决方案 1. Redis分布式锁完整实现 // 原子加锁 防死锁 String uuid UUID.randomUUID().toString(); Boolean locked redisTemplate.opsForValue().setIfAbsent(&qu…

MATLAB 2025a的下载以及安装,安装X310的测试附加功能(附加安装包)

首先将安装包下载到本地中之后解压该文件夹&#xff0c;打开文件发现有两个文件&#xff0c;其中crach文件夹中是破解matlab所用到的文件。而另一个压缩包就是需要安装的文件&#xff0c;要先解压在安装。在安装之前将网络断开&#xff0c;不然可能破解不成功&#xff0c;先进入…

Scala实用编程(附电子书资料)

概述 Scala 是一种多范式编程语言&#xff0c;结合了面向对象编程&#xff08;OOP&#xff09;和函数式编程&#xff08;FP&#xff09;的特性电子书资料&#xff1a;https://pan.quark.cn/s/88737d4a680d Scala 的核心特点多范式融合 既支持面向对象编程&#xff08;类、继承、…

数据结构(8)双向链表

目录 一、概念与结构 二、双向链表的实现 1、初始化 2、尾插 3、头插 4、尾删 5、头删 6、在指定位置之后插入结点 7、删除指定位置的结点 三、完整参考代码 一、概念与结构 这里的双向链表是指带头的的双向循环链表&#xff0c;这里的“带头”和之前所说的“头结…

【DeepSeek-R1 】分词系统架构解析

文章目录 🧩前言 🔍 1. SentencePiece Unigram 的核心原理 1.1 算法基础框架 1.2 核心数学原理 1.3 与BPE/WordPiece的对比 ⚙️ 2. DeepSeek-R1 分词器实现细节 2.1 词表结构设计 2.2 关键特性实现 📊 3. 性能优化关键技术 3.1 加速策略对比 3.2 编码过程伪代码 🔬 4.…

Linux自主实现shell

以下是在Linux操作系统 centos7版本下实现的shell &#xff0c;该shell具备bash的基础功能&#xff0c;无上下键输入历史命令功能&#xff0c;删除字符或命令时按住Ctrl Back #include<stdio.h> #include<stdlib.h> #include<string.h> #include<unistd.…

vue+elementUI上传图片至七牛云组件封装及循环使用

1.效果&#xff08;解决循环组件赋值问题&#xff09; 废话不多说直接上代码 2.下载七牛云依赖 npm install qiniu-js # 或者使用 yarn yarn add qiniu-js3.在vue组件中引入 import * as qiniu from qiniu-js4.在components文件夹下创建UploadImg1/uploadImg.vue组件 <templ…

2025年6月电子学会青少年软件编程(C语言)等级考试试卷(一级)

答案和更多内容请查看网站&#xff1a;【试卷中心 -----> 电子学会 ----> C/C ----> 一级】 网站链接 青少年软件编程历年真题模拟题实时更新 一、编程题 第 1 题 希望如光 题目描述 在充满挑战的生活中&#xff0c;希望往往是支撑人们穿越黑暗的核心力量。这…

拒绝复杂,AI图表制作简单化

在信息爆炸的时代&#xff0c;数据可视化已成为传递信息的核心手段。无论是职场汇报中的业绩分析&#xff0c;还是学术研究里的实验数据呈现&#xff0c;一张清晰直观的图表往往能胜过千言万语。而 AI 技术的介入&#xff0c;彻底改变了图表制作的传统模式 —— 它不仅让零基础…

easypoi生成多个sheet的动态表头的实现

在使用 EasyPOI 导出 Excel 时&#xff0c;生成多个 Sheet 且每个 Sheet 的表头是动态的&#xff08;即每个 Sheet 的列数和列名可能不同&#xff09;&#xff0c;可以通过如下方式实现&#xff1a;✅ 实现原理简述 使用 Workbook workbook ExcelExportUtil.exportExcel(expor…

移除链表元素+反转链表+链表的中间节点+合并两个有序链表+环形链表约瑟夫问题+分割链表

一、移除链表元素 给你一个链表的头节点 phead 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头节点 (列表中的节点数目在范围 [0, 104] 内) 示例&#xff1a;输入&#xff1a;head [1,2,6,3,4,5,6], val 6 …

vue3+arcgisAPI4示例:轨迹点模拟移动(附源码下载)

demo源码运行环境以及配置运行环境&#xff1a;依赖Node安装环境&#xff0c;需要安装Node。 运行工具&#xff1a;vscode或者其他工具。 配置方式&#xff1a;下载demo源码&#xff0c;vscode打开&#xff0c;然后顺序执行以下命令&#xff1a; &#xff08;1&#xff09;下载…

Design Compiler:Milkyway库的创建与使用

相关阅读 Design Compilerhttps://blog.csdn.net/weixin_45791458/category_12738116.html?spm1001.2014.3001.5482 DC Ultra推出了拓扑模式&#xff0c;在综合时会对标准单元进行粗布局(Coarse Placement)并使用虚拟布线(Virtual Routing)技术计算互联延迟&#xff0c;关于拓…

嵌入式教学的云端革命:高精度仿真如何重塑倒车雷达实验与工程教育——深圳航天科技创新研究院赋能新一代虚实融合实训平台

一、嵌入式教学的困境与破局之道 在传统嵌入式系统教学中&#xff0c;硬件依赖始终是核心痛点。以“倒车雷达实验”为例&#xff0c;学生需操作STM32开发板、超声波传感器、蜂鸣器等硬件&#xff0c;面临设备损耗、接线错误、调试效率低等问题。更关键的是&#xff0c;物理硬件…

flutter-boilerplate-project 学习笔记

项目地址&#xff1a; https://github.com/zubairehman/flutter_boilerplate_project/tree/master 样板包含创建新库或项目所需的最小实现。存储库代码预加载了一些基本组件&#xff0c;例如基本应用程序架构、应用程序主题、常量和创建新项目所需的依赖项。通过使用样板代码…