前言:在讲完环境变量后,相信大家对Linux有更进一步的认识,而Linux进程概念到这也快接近尾声了,现在我们了解Linux进程中的地址空间!...本篇主要内容: 了解程序地址空间 理解进程地址空间 探究页表和虚拟地址空间 1....&s.c } 注意:栈区是整体向下增长,局部想上使用的,就是地址最低处,依次往上放后面的元素 但是如果我们将代码更改还能运行过去嘛? char *str = "Hello, Linux!"...首先引入一个概念:每一个进程运行之后,都会有一个进程地址空间的存在,在系统层面都要有自己的页表映射结构! 因此:当一个进程先修改后,它就不再指向原来那块物理空间,而是拥有一个新的物理空间!...结束进程地址空间,我们的Linux进程概念到这里也结束了,后面我将带大家走进进程控制。 谢谢大家支持本篇到这里就结束了
可以通过代码的方式在不同的区域创建变量然后来取地址获取再进行比较 2、栈区和堆区是相对而行的!! ...),然后开辟一块新的物理内存,修改页表的映射 4.2 进程地址空间是如何切换的 进程PCB结构体里有对应的进程地址空间指针,所以进程切换就以为这进程空间地址空间被切换,而页表会被存储在CPU的cr3...这是有Linux的内存模块去管理的,进程并不需要关心。 结论4:其实变量名在定义的时候就已经被转化成一个个虚拟地址了,而我们之所以有a和&a,本质上是为了区分想获取的是变量的值还是地址。...结论5:以前我们所学习的C内存管理,其实本质上是进程地址空间,而内存管理是由Linux替我们完成的,我们上层语言并不需要关心具体的细节,只需要正常去通过对应的线性地址去使用就行了。...六、命令行参数和环境变量在栈的上面 所以环境变量和命令行参数是在栈之上的一个独立空间。
: %p\n", mem); printf("heap addr: %p\n", mem1); printf("heap addr: %p\n", mem2); //临时变量在栈上开辟空间...也就是有 32 条线路,每条线路能通过高低电平的转换来实现0和1的变化,所以这 32 条线路能表示的不同地址组合数量为 2^32个,因为每个内存地址对应一个字节,所以 32 位计算机理论上能直接访问的内存空间大小就是...,类似于bit这样的存储单元是没有地址的概念的 所以所谓的进程地址空间,本质上是一个描述进程可视范围的大小,地址空间内一定要存在各种区域的划分,只要对虚拟地址(线性地址)进行区域划分即可 这里要注意的是...,栈的start是高地址处,其他用户空间都是start为低地址处 3、进程地址空间这样组织的优势 (一)让进程以一个统一的视角看待内存 我们以页表这样的形式用来过渡,保证了我们所访问的虚拟地址(线性地址...物理内存则被划分为与虚拟页大小相同的页框。
Linux进程地址空间是学习Linux的过程中,我们遇见的第一个难点,也是重中之重的重点。虽然它很难,但是,等我们真正懂得了这样设计的原理,我们不禁会感叹:这真的是太妙了。...但是,堆区和栈区的增长的方向是怎样呢?这个好办;方向是比对出来的,我们只需要多申请几次堆空间和栈空间,然后比较地址大小变化。...这里,我还想和大家达成几个共识: 地址空间描述的基本空间大小为字节。 在32位环境下,一共需要2^32个地址。 2^32*1字节=4GB的空间大小。 每个字节都有唯一的地址。 1....虽然每一个进程只有1个虚拟空间,但是这个虚拟空间可以被别的进程来同享。如:子进程同享父进程的地址空间,而mm_user和mm_count就对其计数。...*mm,unsigned long addr); unsigned long mmap_base; unsigned long task_size; //具有该结构体的进程的虚拟地址空间的大小
显示相同地址,却是不同的值 下面在Linux上验证 创建test.c文件 st.c ⮀...,子进程对数据的修改,不影响父进程 子进程要对value修改时,在内存中重新申请一块空间,拷贝value值给新空间,重新映射指向新开辟的空间,导致不影响父进程的value值 ,最终将新开辟的空间value...磁盘中存在很多可执行程序 进程被加载到物理内存中,其中包括了PCB、代码和数据 CPU读取进程的代码和数据按顺序去执行 操作系统加载第二个进程时,在物理内存中申请没有被用到的空间,把PCB、代码和数据放入空间中...,在地址空间中申请空间,在页表处只填写虚拟地址,物理地址处不填写,就不需要在物理地址处申请空间,过一会,进程尝试对空间写入,在重新申请空间把映射关系创建好,整体机制被叫做 缺页中断 8....进程地址空间+页表的意义: 1.防止地址随意访问,保护物理内存与其他进程 若没有地址空间的存在,则直接使用cPU调用物理地址,会有野指针的问题存在 2.将进程管理和内存管理进行解耦合 因为有虚拟地址和页表的存在
; 3.程序地址空间 3.1地址空间简介 我们的程序存储区分为代码区,字符常量区,全局数据区,堆区,栈区等等几个部分,地址是有低地址到高地址进行增长的,我们把这个叫做地址空间; 在栈区定义的变量,这个先定义先入栈...,后定义的变量后入栈,因此这个后定义的变量的地址会更大,这个栈区是向下增长,同理这个,堆区是向上增长的;堆区和栈区相对而生; static修饰的局部变量编译的时候,已经被搞到全局数据区里面了,这个局部变量的生命周期就是一个全局变量的...,所以这个打印的结果是一样的,但是这个实际上的物理地址不是一样的; 3.4谈谈细节 到底什么是进程地址空间:数据总线排列组合形成的地址的范围[0,2^32); 进程地址空间实际上就是我们的进程的一个可以使用的范围...对于任何一个进程,都会创建一个task_struct结构体对象,这个指针指向我们的进程地址空间对象,这个里面就有我们的各种区域的划分,方便我们对于这个区域的管理; 3.6进程地址空间的存在意义 让所有的进程以一个统一的视角去看待内存...,因为地址空间使得所有的进程都需要虚拟地址,子进程和父进程的关系,调度是一套统一的流程; 当我们访问内存的时候,会增加一个转换的过程,在这个转换的过程中,虚拟地址空间会进行审查,例如我们对于这个只读区域进行修改
今天在Linux上调试程序程序的时候发现有时候程序会莫名其妙的hang住,于是就想能不能找到当时程序有那些线程,都在做什么。找了一下linux命令,还真可以满足我的需求。下面看一个小例子。...args=()) t1.start() t2.start() time.sleep(12345) 然后运行这个程序 $ python test.py 先使用 “pstree -apl ” 查看进程结构...in __libc_start_main () from /lib64/libc.so.6 #20 0x000000000040071e in _start () 这里多说一句,如果要看java程序的栈信息
在Linux中,特殊情况,我们将这种地址也成为逻辑地址。 2.感性的理解虚拟地址空间 从前有一个大富翁,他有10亿美元的资产。...3.现象的具体解释 父进程和子进程都有自己独立的进程地址空间,也有独立的页表结构。子进程由父进程创建,因此子进程的进程地址空间是拷贝父进程的进程地址空间。...当然对于Linux而言,虚拟地址、线性地址、逻辑地址都是一样的。 三、怎么办 操作系统要为每一个进程分配地址空间,那么操作系统是否要管理这些地址空间呢?当然是要管理的。...(PCB中的一个属性mm_struct) 区域划分和调整 地址空间有很多区域:栈区、堆区、数据段、代码段等,那么进程地址空间是如何进行区域划分的呢?...本文作者目前也是正在学习Linux相关的知识,如果文章中的内容有错误或者不严谨的部分,欢迎大家在评论区指出,也欢迎大家在评论区提问、交流。
Linux进程地址空间 零、前言 一、程序内存空间 二、进程地址空间 1、引入及概念 2、进程地址空间 3、相关问题 零、前言 本章主要讲解学习进程地址空间的知识 一、程序内存空间 在学习C/C...++中我们知道了程序内存的空间开辟以及内存分区的基本概念 示图: 各分区作用: 内核空间:用户代码无法读写 命令行参数环境变量:储存命令行参数环境变量 栈区:存放运行函数而分配的局部变量...,说明该地址绝对不是物理地址,因为是物理地址根本不会有这种事发生 2、进程地址空间 概念: 在Linux地址下,这种地址叫做 虚拟地址,我们在用C/C++语言所看到的地址,全部都是虚拟地址!...,随进程的创建而创建,随进程的退出而回收 进程地址空间的内容: 进程地址空间是由0x00000000到0xffffffff的线性地址空间,按照刻度被划分为各个区域,例如代码区、堆区、栈区等...在结构体mm_struct当中,便记录了各个边界刻度(对于堆向上增长以及栈向下增长实际就是改变mm_struct当中堆和栈的边界刻度) 示图: 注:在结构体mm_struct中各个刻度之间的每一个刻度都代表一个虚拟地址
今日更新了Linux进程空间地址和进程调度队列的内容 欢迎大家关注点赞收藏⭐️留言 问题现象 当g_val的值修改后,父子进程各自的g_val地址都是一样的,但是为什么值却不同呢?...进程地址空间 地址空间的本质就是内核中的一个结构体对象。 每一个进程都有自己独立的地址空间,都有自己独立的页表。 页面里面存的是虚拟地址和对应的物理地址。...由于g_val的值改了,由因为进程具有独立性,所以OS就会在物理空间重新开一个空间,拷贝旧的数据到该地址,再让子进程的g_val指向该物理地址,但是虚拟地址不变。因此就出现文章开头的情况。...如果父子进程都不写,他们的变量默认是被父子共享的,代码是共享的,物理空间不会分离。 这种拷贝也叫写时拷贝,即按需申请,可有效节省空间。...Linux2.6内核进程调度队列 上图是Linux2.6内核中进程队列的数据结构。
】进程理解与学习Ⅰ-进程概念 浅谈Linux下的shell--BASH 【Linux】进程理解与学习Ⅱ-进程状态 【Linux】进程理解与学习Ⅲ-环境变量 ---- 前言...进程地址空间 前文回顾 首先,我们先来回顾一下,在指针阶段我们学习了,内存被划分为一个一个内存单元,每一个单元的大小为1字节。而每一个内存单元都有自己的编号,从0-0xFFFFFFFF。...在Linux中我们称之为虚拟地址/线性地址。 OS则是将虚拟地址转化为物理地址(如何转化后面会讲到) 如何理解进程地址空间? 首先我们要知道,什么是进程地址空间?...实际上进程地址空间就是操作系统喂给进程的一块“饼”,OS会跟每个进程说,你们有4G的内存空间(栈区、堆区、静态区...)可以使用,但实际上,只有当进程需要用的时候,OS才会分配空间给进程。...接下来谈一谈OS如何管理我们所说的进程地址空间(即我们所说的栈区、堆区等)? 答:先描述,再组织。实际上我们所说的进程地址空间本质上是一个内核数据结构,struct_mmstruct{}。
进程栈的初始化大小是由编译器和链接器计算出来的,但是栈的实时大小并不是固定的,Linux 内核会根据入栈情况对栈区进行动态增长(其实也就是添加新的页表)。...【扩展阅读】:如何确认进程栈的大小 我们要知道栈的大小,那必须得知道栈的起始地址和结束地址。栈起始地址 获取很简单,只需要嵌入汇编指令获取栈指针 esp 地址即可。...然而,如果达到了最大栈空间的大小,就会发生 栈溢出(stack overflow),进程将会收到内核发出的 段错误(segmentation fault) 信号。...二、线程栈 从 Linux 内核的角度来说,其实它并没有线程的概念。Linux 把所有线程都当做进程来实现,它将线程和进程不加区分的统一到了 task_struct 中。...线程仅仅被视为一个与其他进程共享某些资源的进程,而是否共享地址空间几乎是进程和 Linux 中所谓线程的唯一区别。
查看整台服务器的硬盘使用情况 cd / 进入根目录 du -sh * 查看每个文件夹的大小
环境变量通常具有某些特殊用途,还有在系统当中通常具有全局特性 1.2 常见环境变量 PATH : 指定命令的搜索路径 HOME : 指定用户的主工作目录(即用户登陆到Linux系统中时,默认的目录)...2.程序地址空间 2.1 研究背景 kernel 2.6.32 32位平台 2.2 程序地址空间回顾 在C语言的时候,有这样的空间布局图 可是我们对他并不理解 来段代码感受一下 #include...在Linux地址下,这种地址叫做 虚拟地址 我们在用C/C++语言所看到的地址,全部都是虚拟地址!...物理地址,用户一概看不到,由OS统一管理 OS必须负责将 虚拟地址 转化成 物理地址 2.3 进程地址空间 所以之前说‘程序的地址空间’是不准确的,准确的应该说成 进程地址空间 说明: 上面的图就足矣说名问题...3.Linux2.6内核进程调度队列 上图是Linux2.6内核中进程队列的数据结构 3.1 一个CPU拥有一个runqueue 如果有多个CPU就要考虑进程个数的负载均衡问题 3.2 优先级
上次介绍了环境变量:Linux:进程概念(四.main函数的参数、环境变量及其相关操作) 1.程序地址空间 牵扯到内存,肯定有事这张图啦。...栈区: 栈区用于存储函数的参数值、局部变量和函数调用返回地址等信息。 栈区是向下增长的,即栈顶地址逐渐减小,整体比堆区的地址要高。 上述空间排布结构是在内存吗?...对于32位系统,进程地址空间通常是从0到4GB,这个范围内包含了代码段、数据段、堆、栈等部分,用于存放程序的指令、数据以及动态分配的内存(就是我们上面那个图) 每个进程都有自己独立的地址空间,使得进程间可以互相隔离...Linux系统中,任何进程最终执行完毕后都会返回一个状态码,这个状态码通常被称为“退出码”或“返回码”(exit code)。...复制父进程数据:新创建的子进程是父进程的副本,所以操作系统会复制父进程的部分数据结构内容到子进程,包括代码、数据、堆、栈等内容。
一,内存地址空间 1.1,栈区 存储变量:普通局部变量,指针变量,函数参数,函数返回地址-,临时变量,寄存器变量; 函数参数:函数的参数是从右到左依次入栈的; 在vs2022上栈区并不是"向下生长的",...动态分配:程序运行时可根据需要随时在堆区申请和释放内存空间,大小可在运行时确定,适应不同数据结构的和算法的需求; 空间较大:一般来说,堆区的空间比栈区的空间更大,因为它是随机开辟的,并不像栈区地址都是连续的...,比如,父子进程执行代码就是使用的同一份代码,因为代码段是可读的,不会发生写实拷贝;所以代码段时是共享的; 3.固定性:这个说的是代码段的大小在程序编译链接时就已经确定,程序中的指令和常量数量固定,决定了代码段的大小...,就会触发写实拷贝机制,重新开辟一块空间,存储新的数据,并且修改页表映射; 4.2进程地址是如何被切换的 进程PCB结构体里有对应的进程地址空间指针,所以进程切换就以为这进程空间地址空间被切换,...PCB; 2.虚拟地址可以相同,但是通过页表映射的物理地址是不同的;各自有各自的区域,对于父子进程子啊写实拷贝的机制下,也是拥有自己独一份的物理地址; 五,命令行参数和环境变量在栈的上面
一、关于进程地址空间的简单理解 进程地址空间其实是分了很多个区域的,区域划分的本质就是区域内的各个地址都是可以使用的。...如同下面这个图所示: 无论是环境变量的地址还是环境变量表的地址,所存放的地址都在栈的上部。这里的已初始化数据和未初始化数据是指的全局变量,包括静态变量(静态变量默认被初始化为0)。...进程地址空间不是真实的物理内存,叫做虚拟内存。每一个进程都有自己独立的PCB,也有自己独立的地址空间。在32位机器下,进程地址空间的大小为[0,4GB]。...其中,PCB会记录一个进程的起始地址或基地址,这其实就是进程地址空间的首地址。...所以,进程地址空间就是数据结构,具体到进程中,就是特定数据结构的对象! 二、为什么要有进程地址空间和页表 1、将物理内存从无序变成有序,让进程以统一的视角看待内存。
首先看linux进程在32位处理器下的虚拟空间内存布局,以i386 32位机器为例 x86_32 32位处理器进程虚拟地址空间布局 每个用户进程的虚拟地址空间为...— 0XC0000000为栈的空间,其中两个区间之间都有一个缝隙。...所以以现在的物理内存大小对于虚拟地址空间完全够用。...linux在x86_64下的经典布局如下图 x86_64 64位处理器进程地址空间布局 用户空间分区一致,区别就是地址空间变大了,内核空间取消了高端内存,因为内核空间的地址空间完全可以访问全部物理内存...下面以32位处理器为例看linux内核如何建立用户进程空间的内存布局的,fork调用是复制父进程的struct mm_struct的内存描述符不需要重新建立布局,而建立新的内存布局是通过加载二进制可执行文件
深入学习进程地址空间并克服Linux学习的第一道险关:4.1中的3:统一性! 1....2.3 地址空间的区域划分 在此之前,大家要确定一个共识: 地址空间描述的基本空间大小是字节。 32位下,有2的32次方个地址。...,与我们所举的例子是一模一样的,就是通过改变边界的大小,从而实现地址空间的动态分配!...而我们知道堆和栈都有可能进行空间的扩大(malloc, new),本质上就是修改各个区域的start或者end。free空间的时候也就对应着缩小stack/heap。...地址空间的存在,可以更方便的进行进程和进程的数据代码的解耦,保证了进程独立性这样的特征。 3:保证统一性(最难点) 接下来就要引入第三点,这一点也是从初学Linux到现在所碰到的第一个难关!
领取专属 10元无门槛券
手把手带您无忧上云