learn_C_deep_1 (C程序补充知识、变量的声明和定义、声明和定义的区别)

简介: learn_C_deep_1 (C程序补充知识、变量的声明和定义、声明和定义的区别)

C程序补充知识


       先让我们来看一段C语言的代码

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<windows.h>
int main()
{
  printf("hello world!\n");
  system("pause");
  return 0;
}


相信我们有过C语言基础的人第一次写的代码都是hello world,本程序也是非常的简单,运行结果就是在终端上输出一句hello world,但是我们有没有真正理解到这段代码后面的知识呢?它是如何运行的,运行的时候加载到电脑的哪一个部分呢?接下来,让我们一探究竟。


这段程序在我们看来首先它是一段文本形式的代码,当我们在集成开发环境中运行它的时候,它就形成了一个可执行的二进制文件,也就是一个可执行的程序。我们可以发现,我们在创建代码的文件中有一个Debug文件,点进去我们会发现有一个应用程序.exe(如下图),这就是我们的可执行程序。它就类似于我们的QQ、微信,双击就可以运行。


我们所有都知道,应用需要双击才可以打开,但是我们有没有思考过,为什么我们的程序要双击呢?


应用程序双击是一种用户与计算机之间的交互方式,本质上是一种操作系统的功能。当用户双击应用程序图标时,操作系统会寻找该程序的可执行文件,然后执行该文件,将程序加载到内存中,并开始运行。


从而我们可以得出:任何程序在执行前都必须加载到内存当中


这里我提出两个问题:


       1.程序没有加载到内存之前,存放在什么位置? -  硬盘


程序在没有被加载到内存中之前,通常是存储在计算机的硬盘或其他存储介质上的。具体来说,程序文件通常被存储为二进制文件或者可执行文件的形式,包含了编译后的代码和数据等信息。


当用户运行程序时,操作系统会从硬盘或其他存储介质中找到该程序文件,并将其加载到内存中,然后将CPU的指针设置为该程序的入口点,使其开始执行。在程序运行期间,CPU会不断地从内存中读取指令和数据,对其进行处理,然后将结果写回内存。


需要注意的是,操作系统并不是直接将整个程序文件加载到内存中,而是进行了分段加载,将程序分成了多个段(例如代码段、数据段、堆栈段等),然后逐个将这些段加载到内存中。这样可以提高内存的利用率,并且使得程序可以更加灵活地使用内存。


       2.为什么程序一定要加载到内存当中呢? -  访问速度快


在计算机执行程序时,需要将程序代码和数据加载到内存中,然后才能让计算机执行这些程序操作。这是因为内存具有比硬盘更快的读取速度,因此将程序加载到内存中可以使计算机更快地执行程序。


另外,程序在运行时需要不断地访问和修改存储在内存中的数据,如果程序没有被加载到内存中,它无法进行这些操作。


因此,将程序加载到内存中是程序能够在计算机上运行的必要前提。 当程序被加载到内存中时,操作系统会为该程序分配一定的内存空间,以及为程序所需的所有库文件和系统资源分配内存空间,以确保程序在运行时可以访问到这些资源。当程序运行结束后,操作系统会释放这些内存空间,以便其他程序使用。


这里需要我们理解一下冯诺依曼体系,先来看一张图片

输入设备:是指用于将外部数据输入到计算机系统中的各种设备。常见的输入设备包括:鼠标、键盘、触摸屏、扫描仪、数码相机、麦克风、手写板等。这些设备可以将不同形式的数据输入到计算机中,如文本、图像、音频、视频等。


存储器:是计算机系统中用于存储数据和指令的硬件设备。它可以分为两种类型:主存储器(也称为随机存取存储器或RAM)和辅助存储器(也称为二级存储器或外存储器)。


主存储器通常是计算机系统中最重要的一种存储器,用于存储当前正在执行的程序和数据。它是由一组芯片组成的,可随时访问其中的任何位置。主存储器的容量通常以字节为单位计算,并以兆字节(MB)或吉字节(GB)为单位表示。


辅助存储器通常是计算机系统中可移动的存储设备,例如硬盘、光盘、U盘、磁带等。它们通常比主存储器容量更大,但使用速度会相对较慢。辅助存储器的数据可以在需要时读取到主存储器中进行处理。


运算器:是计算机的主要组成部分之一,也称为算术逻辑单元(ALU)。它是一种专门用于执行算术和逻辑操作的硬件设备。运算器通常包括多个算术逻辑单元(ALU),用于执行各种算术运算,例如加、减、乘、除等,以及逻辑运算,例如与、或、非、异或等。


控制器:是计算机的一个重要组成部分,它是一个电子电路,负责协调计算机内部各部分的运作,以及控制计算机的操作和处理过程。


控制器主要包括两部分:指令寄存器和指令译码器。指令寄存器用于存储CPU当前正在执行的指令,指令译码器则用于解析指令并将其转换成CPU可以执行的操作码。当CPU需要执行一条指令时,它会从主存储器中读取该指令,并将其存储到指令寄存器中。然后,指令译码器会解析该指令,将其转换成一系列操作码,并向各个部件发出控制信号,协调各部件的运作,以完成指令的执行过程。除此之外,控制器还负责处理各种中断信号,例如外部设备的输入输出请求、硬件故障等,以保证计算机的稳定运行。


输出设备:是计算机系统用于向用户显示计算结果、输出信息和数据的设备,它可以将计算机处理过的数据和结果进行可视化或声音化等形式输出。 常见的输出设备包括:  显示器 、打印机、 喇叭或扬声器 、投影仪等


当我们从输入设备(键盘)输入数据时,然后编写好代码,将代码文件经过编译器编译后会生成可执行文件,数据首先会被放入到存储器(硬盘),当我们双击可执行文件时,操作系统会将该文件加载到内存中。加载时,操作系统会分配一段内存空间给程序,为代码文件分配虚拟内存地址,当CPU执行程序时,它会从内存中读取程序代码和数据,并在CPU内部执行指令。CPU执行指令时,它会将结果存储在内存中,也可将结果输出到显示器、打印机等输出设备上。  当程序执行完毕后,操作系统会释放分配给程序的内存空间,并将程序从内存中卸载。这就是我们hello world的整个程序执行过程啦!


变量的声明和定义


这里依然提出几个问题:


1.什么是变量?


在计算机编程中,变量是指一个名称代表的内存位置,用于存储程序运行时需要的数据。变量可以存储各种类型的数据,例如数字、字符串、布尔值等。

变量的名称由程序员定义,并根据需要指定变量的数据类型。变量的数据类型决定了变量可以存储的数据类型和占用的内存空间大小。

例如,整数类型的变量可以存储整数值,而字符串类型的变量可以存储字符串值。

在程序运行过程中,程序可以不断地给变量赋值,并将变量的值用于计算、比较或其他操作。变量的值可以改变,但其内存位置不变。

因此,程序可以通过变量来存储和访问各种数据,并通过数据的改变来实现程序的逻辑。 总之,变量是用于存储程序运行时需要的数据的一种机制,可以存储各种类型的数据,并通过其名称和数据类型来在程序中访问和操作数据


2.变量的本质是什么? -  所有的变量都要在内存的某个位置开辟空间


在计算机内部,变量本质上是一段内存空间用来存储数据。 通过变量,我们可以给这段内存空间一个名称,方便我们在程序中引用和修改这段内存中的数据。


3.变量的定义和声明形式、初始化和赋值的区别


类型 变量名 = 默认值; -   定义


类型 变量名; -   声明

int a;//声明
double d = 20.0;//定义
char c = 'c';//初始化
c = 'a';//赋值


变量初始化是指在定义一个变量时,为它赋予一个初始值。在大多数编程语言中,变量的初始值默认是一些特定的值,例如数值变量的初始值为0,布尔型变量的初始值为false。如果在定义变量时明确赋予了一个初始值,则我们称之为变量初始化。

变量赋值是指在程序执行过程中,为一个已经存在的变量重新赋值。我们可以将一个新的值赋予一个变量,也可以将一个变量的值赋予另一个变量,从而更新这个变量的值。

总的来说,变量初始化是在定义变量时为其赋予一个初始值,而变量赋值是在程序执行过程中更改一个已经存在的变量的值。


4.为什么要定义变量


定义变量的主要目的是为了在程序中存储和处理数据。在许多计算机程序中,我们需要存储和操作各种类型的数据,例如数字、字符串、布尔值等等。如果没有变量,我们就无法在程序中存储和引用这些数据。

通过定义变量,我们可以在程序中给这些数据赋予一个名称或者标识符。这个标识符可以表示一个具体的值,而且这个值可以被程序随时更新和修改。这样,我们可以在程序中灵活地使用这些数据,根据不同的情况来更新和处理数据。同时,使用变量还能提高程序的可读性和可维护性,让代码更加易于理解和修改。


定义变量还可以用来节约内存空间。当我们需要使用一些较大的数据时,可以创建一个变量来存储这些数据,而不是一直重新创建。这样可以减少内存的开销,让程序更加高效。而且在程序中的数据并不是目前需要使用,因此我们需要变量去存储它。


程序运行:需要加载到内存当中


程序计算:需要使用变量


定义变量的本质:为数据在内存中分配一段存储空间,通过这个存储空间的地址来访问和修改数据。


声明和定义的区别


定义和声明是编程中两个重要的概念,它们有一些相似之处,但也有不同。

声明通常是指在代码中告诉编译器某个变量、函数等的存在和基本信息,而不进行具体实现。在声明时,我们只需要指定变量的类型和名称即可,不需要为其分配内存空间或者进行其他操作。


举个例子,如果我们要在C语言中使用一个名为x的整型变量,我们可以这样声明它:


```int x;```


在这个声明中,我们只是告诉编译器x是一个整型变量,在后续的代码中会用到它。但是这个声明并没有分配内存空间或者进行其他操作,因此x的具体值是未知的。


定义则是指在代码中为变量或者函数等分配内存空间,并且进行相应的初始化或者实现。在定义时,我们需要为变量分配内存空间,为其赋予初始值或者进行其他操作。


回到上面的例子,我们如果需要定义这个变量,就需要在声明的基础上进行具体实现:


```int x = 0;```


在这个定义中,我们为x分配了一个整型的内存空间,并将其初始值设为0。这样,在后续的代码中,我们就可以使用这个变量并对它进行操作了。


总的来说,声明和定义都是为了告诉编译器程序中的某个变量或者函数的存在和基本信息,但声明不会分配内存空间或者进行其他操作,而定义会在声明的基础上进行具体实现,分配内存空间,并进行初始化或者实现。


这里我们用个小故事来讲解其区别


       寝室的四个人都喜欢一个名叫小花的女同学,寝室里面有一个人最大胆,率先向小花说做我女朋友,小花也说了我愿意。为了防止寝室其他人再去向小花表白,此时你肯定就会跑到寝室说小花已经是我女朋友啦,你们要保持好距离,寝室其余人听完都默默的伤心起来。


在这个故事里:表白小花就是定义,只能定义一次,向舍友告知就是声明,声明可以有多次。

相关文章
|
8月前
|
存储 安全 Java
解释Python中的引用和赋值机制。
Python中,变量是对象引用,不存储数据,而存数据在内存的位置。赋值(=)创建变量并让其指向内存中的对象。当多个变量指向同一对象时,它们共享引用。Python使用引用计数管理对象生命周期,对象引用为0时回收。了解这些机制对优化内存使用和防止内存泄漏很重要。例如: ```markdown ```python a = 5 b = a # b引用了同一数字5 del a # 数字5的引用计数仍为1,未被回收 ``` 引用计数并非唯一机制,Python还采用其他策略处理循环引用等复杂情况。
66 2
|
8月前
|
编译器 程序员 C++
在C++语言中函数的声明
在C++语言中函数的声明
60 0
|
JavaScript 前端开发
什么是函数?函数分为几种,如何声明?区别是什么?
什么是函数?函数分为几种,如何声明?区别是什么?
93 0
|
编译器 Python
python之局部变量和全局变量的定义,两者之间的区别和使用方法,global和nonlocal的定义和使用方法,可变与不可变类型的定义和示例
python之局部变量和全局变量的定义,两者之间的区别和使用方法,global和nonlocal的定义和使用方法,可变与不可变类型的定义和示例
172 0
|
3月前
|
存储 编译器 C语言
C语言函数的定义与函数的声明的区别
C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。
|
3月前
|
编译器 C语言
变量的声明与定义区别
变量的声明是指预先告知编译器变量的名称和类型,但不分配内存;而定义则是声明的同时在内存中分配空间,可以初始化。简单来说,声明是告诉编译器“有这么一个东西”,定义是“创建并使用这个东西”。
135 11
|
5月前
【函数】函数的声明和定义
【函数】函数的声明和定义
110 1
|
8月前
|
存储 程序员 Shell
【C/C++ 基本语法 结构体】C++ 结构体声明(定义)以及不同写法的差异
【C/C++ 基本语法 结构体】C++ 结构体声明(定义)以及不同写法的差异
122 1
|
8月前
|
编译器 C++
在C++语言中类的定义和声明
在C++语言中类的定义和声明
60 0
|
8月前
|
存储 Java Python
Python 变量?对象?引用?赋值?一个例子解释清楚
Python 变量?对象?引用?赋值?一个例子解释清楚