学过编程都知道,在所有高级编程语言中,C是运行最高效的语言, 一个重要原因是C语言是静态型语言,会直接被编译成机器码,然后被计算机直接运行。和大部分人一样,做对C语言编译之后的运行方式的充满了好奇,主要包括一下几个方面:

可执行程序是什么,里面只是包括一大堆机器码吗?

C语言中定义变量, 常量, 函数等等,在内存中是如何分布的,它们之间是靠什么相互访问的?

计算机当中,同时运行的进程非常多,每个进程都占用内存空间,程序编译的时候,如何知道自己运行时将要被加载到哪里,又如何被CPU执行?

下面是一段简单的C程序:

#include

int a = 10;

int main(){

printf("The a is %d\n",a);

return 0;

}

用gcc编译, 运行会打印出”The a is 10″,这就是一个可执行文件,查资料得知linux下可执行文件为elf格式他有三种不同的类型:可重定位的目标文件、可执行文件和共享库。用readelf工具查看次文件的头信息:

[root@hitoy ~]# readelf -h a.out

ELF Header:

Magic: 7f 45 4c 46 01 01 01 00 00 00 00 00 00 00 00 00

Class: ELF32

Data: 2’s complement, little endian

Version: 1 (current)

OS/ABI: UNIX – System V

ABI Version: 0

Type: EXEC (Executable file)

Machine: Intel 80386

Version: 0x1

Entry point address: 0x8048310

Start of program headers: 52 (bytes into file)

Start of section headers: 1980 (bytes into file)

Flags: 0x0

Size of this header: 52 (bytes)

Size of program headers: 32 (bytes)

Number of program headers: 8

Size of section headers: 40 (bytes)

Number of section headers: 30

Section header string table index: 27

所以可以得出结论,可执行文件存放的不只是可执行的机器码,还包括其它一些说明和帮助程序运行的信息,linux下一个可执行的elf文件的结构如下:

ELF可执行文件格式

ELF Header描述了体系结构和操作系统等基本信息,并指出Section Header Table和Program Header Table在文件中的什么位置, Program Header Table记录文件里segment的分布,segment里存放的是各个数据段(section)。Section Header Table中保存了所有Section的描述信息,通过Section Header Table可以找到每个Section在文件中的位置。在上面的ELF Header中,有一个入口地址(Entry point address),当程序运行时,CPU从这里开始执行,程序中不仅仅保存有入口地址,程序中的其它数据在编译之后都有一个确切的地址,这样CPU就可以访问了。

那么问题又来了,程序在编译的时候,并不确定自己将要在哪里运行,运行的环境可能还各不一样,例如有些计算机只有512MB内存,有些可达到4G,里面的地址管用吗?而且我们注意到,里面的地址都是线性连续的,内存刚好能分配这样一段空间?把上面的C程序中全局变量a的值改成11,重新编译,发现执行的入口地址和原来的一样!

实际上操作系统主要的功能之一就是内存管理, 现代操作系统普遍采用虚拟内存管理(Virtual Memory Management)机制,这需要处理器中的MMU(Memory Management Unit,内存管理单元)提供支持, 有了MMU的支持,内存地址可以划分为虚拟地址和物理地址, 当存在MMU支持时,CPU所有的寻址操作都会由MMU翻译成实际的物理地址,然后操作实际的内存。有了MMU,就可以保证程序中的地址连续,而不用关心实际数据在内存里的存放位置了, 也解释了为什么不一样的程序入口地址是一样的。

MMU工作原理

那么对于不同配置的计算机,多个程序同时执行时,这些地址够用吗?通常操作系统把虚拟地址空间划分为用户空间和内核空间, 对于同样的平台,它们的大小通常是确定了的, 例如x86平台的Linux系统虚拟地址空间是0x00000000~0xffffffff,前3GB(0x00000000~0xbfffffff)是用户空间,后1GB(0xc0000000~0xffffffff)是内核空间。也就是说不论你的x86机器内存是多少,对于一个程序来说,它的地址空间可以为3G,操作系统会根据内存的使用情况,进行动态管理。

参考资料

编译和执行区别 c语言,C语言编译和执行分析相关推荐

  1. python中动态语言静态语言的定义_作为程序开发,你所需要知道的编译型与解释型、动态语言与静态语言、强类型语言与弱类型语言的概念以及区别...

    作为程序开发,你所需要知道的编译型与解释型.动态语言与静态语言.强类型语言与弱类型语言的概念以及区别! 在各式各样的开发过程当中,我相信各位开发小伙伴在开发过程中并没有太关注什么是解释性语言和编译性语 ...

  2. 【Groovy】Groovy 动态语言特性 ( Groovy 语言与 Java 语言执行效率对比 | 以动态特性编译的 Groovy 类 | 以静态特性编译的 Groovy 类 )

    文章目录 一.以动态特性编译的 Groovy 类 二.Groovy 语言与 Java 语言执行效率对比 三.以静态特性编译的 Groovy 类 一.以动态特性编译的 Groovy 类 Groovy 类 ...

  3. c语言中编译解释,C语言既可以编译执行又可以解释执行吗? 编译执行怎么解释? 解释执行又怎么解释?...

    C 语言程序仅可以解释执行. 解释程序是将源程序(如BASIC)作为输入,解释一句后就提交计算机执行一句,并不形成目标程序.编译程序是把高级语言(如FORTRAN.COBOL.Pascal.C等)源程 ...

  4. c语言编译器怎窗口怎么执行,c语言编译(c语言编译执行详解)

    大家一般都用的是什么软件呢? 1.C语言编程软件有哪些:Mcrosoft Visual C++ .Microsoft Visual Studio. DEV C++.Code::Blocks.Borla ...

  5. c语言程序只能编译执行吗,c语言的源程序不必通过编译就可以执行对吗

    错误,C语言采用编译方式将源程序转换为二进制的目标代码,编写好一个C程序到完成运行一般经过以下几个步骤,编辑和编译,就是将已经编辑好的源程序翻译成二进制的目标代码,经编译后的得到的二进制代码还不能直接 ...

  6. c语言源程序不编译也能直接运行吗,c语言的源程序不必通过编译就可以执行对吗...

    错误,C语言采用编译方式将源程序转换为二进制的目标代码,编写好一个C程序到完成运行一般经过以下几个步骤,编辑和编译,就是将已经编辑好的源程序翻译成二进制的目标代码,经编译后的得到的二进制代码还不能直接 ...

  7. c与python的区别-Python与C语言有什么区别?

    答题练手.手机排版不佳请谅解~ 更新2:还是关于编译和解释 二者的本质区别是在编译/解释器的总体工作方式上的,编译器是off-line,解释器是on-line.编译器把整个程序读进来,进行一系列变大变 ...

  8. python与c语言在语法上的区别-python和c语言的区别是什么

    Python可以说是目前最火的语言之一了,人工智能的兴起让Python一夜之间变得家喻户晓,Python号称目前最最简单易学的语言,现在有不少高校开始将Python作为大一新生的入门语言.本萌新也刚开 ...

  9. [GO语言基础] 二.编译运行、语法规范、注释转义及API标准库知识普及

    作为网络安全初学者,会遇到采用Go语言开发的恶意样本.因此从今天开始从零讲解Golang编程语言,一方面是督促自己不断前行且学习新知识:另一方面是分享与读者,希望大家一起进步.前文介绍了什么是GO语言 ...

  10. 关于HotSpot VM以及Java语言的动态编译 你可能想知道这些

    目录 1 HotSpot VM的历史 2 HotSpot VM 概述 2.1 编译器 2.2 解释器 2.3 解释型语言 VS 编译型语言 3 动态编译 3.1 什么是动态编译 3.2 HotSpot ...

最新文章

  1. Laravel - Artisan 个人常用总结
  2. 基于FPGA的超声波数据图像显示
  3. js(Dom+Bom)第六天(1)
  4. 列举python的5个数据类型_python公开课|新公布的5个python核心数据类型,这些细节你难道还不不知道吗...
  5. 带注释源码php,php的注释方法
  6. c 语言 数据库 pdf下载,Visual C/C++ 编程精选集锦 数据库及图形图像分册 PDF扫描版[38MB]...
  7. R语言高级算法之支持向量机(Support Vector Machine)
  8. ECCV 2016 paper list
  9. SVN服务器搭建--Subversio与TortoiseSVN的配置安装(Windows)
  10. FPGA实验记录一:1位全加器设计
  11. 我是怎样开发一个开源系统的安全模块?
  12. ArcMap符号库 .style 和 .ServerStyle 制作过程:(对两区划定实例分析)
  13. 通过docker创建Nginx容器并运行Vue项目(可用https进行访问)
  14. 机器学习笔记 - MediaPipe了解 + 结合OpenCV进行人体姿势估计
  15. python 什么意思_Python中冒号等于(:=)是什么意思?
  16. oracle实例恢复 redo,ORACLE不完全恢复之current或active状态redo损坏(二)
  17. 2023二建学天案例突破101问
  18. 微信公众号Python开发(Wechatpy+新浪云SAE应用)
  19. 中学计算机教师招聘面试题,中学教师招聘面试题目-20210528215022.docx-原创力文档...
  20. MM配额协议(Quota Arrangement)-阿龙学习MM PA 笔记(3)

热门文章

  1. linux下怎么批量命名文件,linux下的文件操作——批量重命名
  2. Unet(pytorch)——制作自己的数据集(二分类)
  3. WinServer 2019 组策略开启远程桌面(增改)
  4. Android绕过微信包名限制对接微信登录和支付
  5. uniapp 上传图片 + 预览图片 + 删除图片
  6. “一万小时定律”:只要你在任何一件事情上花1万小时来练习,就会成为大师[转载]
  7. 入门深度学习?这里有5件你应该知道的事
  8. 产品研究:WPS如何在Office的“围剿”下突围
  9. 手机 CPU 架构类型了解
  10. 高斯消元法解线性方程组数学原理及Python手工实现