c语言章节6

本文目录

在前面一节中简单介绍了变量的使用,当我们定义一个变量的时候,系统就会为变量分配一块存储空间。而变量的数值在内存中是以二进制的形式存储的,这讲来深入研究变量在内存中的一些存储细节。

回到顶部

一、字节和地址

为了更好地理解变量在内存中的存储细节,先来认识一下内存中的“字节”和“地址”。

1.计算机中的内存是以字节为单位的存储空间。内存的每一个字节都有一个唯一的编号,这个编号就称为地址。就好像酒店是以房间为单位的,每个房间都有一个唯一的房号,我们根据房号就能找到对应的房间。

里面的每个小框框就代表着内存中的一个字节,白色数字就是每个字节的地址(这里采取十六进制来显示,地址值是随便写的,仅作为参考,真实情况中的地址值不一定是这个),可以发现,内存中相邻字节的地址是连续的。

2.大家都知道,一个字节有8位,所能表示的数据范围是非常有限的,因此,范围较大的数据就要占用多个字节,也就是说,不同类型的数据所占用的字节数是不一样的。

回到顶部

二、变量的存储

1.变量类型的作用

跟其他编程语言一样,C语言中用变量来存储计算过程使用的值,任何变量都必须先定义类型再使用。为什么一定要先定义呢?因为变量的类型决定了变量占用的存储空间,所以定义变量类型,就是为了给该变量分配适当的存储空间,以便存放数据。比如char类型,它是用来存储一个字符的,一个字符的话只需要1个字节的存储空间, 因此系统就只会给char类型变量分配1个字节,没必要分配2个字节、3个字节乃至更多的存储空间。

2.变量占用多少存储空间

1> 一个变量所占用的存储空间,不仅跟变量类型有关,而且还跟编译器环境有关系。同一种类型的变量,在不同编译器环境下所占用的存储空间又是不一样的。我们都知道操作系统是有不同位数的,比如Win7有分32位、64位,编译器也是一样的,也有不同位数:16位、32位、64位(Mac系统下的clang编译器是64bit的)。由于我们是Mac系统下开发,就以64位编译器为标准。

2> 下面的表格描述了在64位编译器环境下,基本数据类型所占用的存储空间,了解这些细节,对以后学习指针和数组时是很有帮助的。

3> 下面的表格描述了在不同编译器环境下的存储空间占用情况

3.变量示例

当定义一个变量时,系统就会为这个变量分配一定的存储空间。

1 int main()
2 {
3     char a = ‘A‘;
4
5     int b = 10;
6
7     return 0;
8 }

1> 在64bit编译器环境下,系统为变量a、b分别分配1个字节、4个字节的存储单元。也就是说:

  • 变量b中的10是用4个字节来存储的,4个字节共32位,因此变量b在内存中的存储形式应该是0000 0000 0000 0000 0000 0000 0000 1010。
  • 变量a中的‘A‘是用1个字节来存储的,1个字节共8位,变量a在内存中的存储形式是0100 0001,至于为什么‘A‘的二进制是这样呢,后面再讨论。

2> 上述变量a、b在内存中的存储情况大致如下表所示:

(注:"存储的内容"那一列的一个小格子就代表一个字节,"地址"那一列是指每个字节的地址)

  • 从图中可以看出,变量b占用了内存地址从ffc1~ffc4的4个字节,变量a占用了内存地址为ffc5的1个字节。每个字节都有自己的地址,其实变量也有地址。变量存储单元的第一个字节的地址就是该变量的地址。变量a的地址是ffc5,变量b的地址是ffc1。
  • 内存寻址是从大到小的,也就是说做什么事都会先从内存地址较大的字节开始,因此系统会优先分配地址值较大的字节给变量。由于是先定义变量a、后定义变量b,因此你会看到变量a的地址ffc5比变量b的地址ffc1大。
  • 注意看表格中变量b存储的内容,变量b的二进制形式是:0000 0000 0000 0000 0000 0000 0000 1010。由于内存寻址是从大到小的,所以是从内存地址最大的字节开始存储数据,存放顺序是ffc4 -> ffc3 -> ffc2 -> ffc1,所以把前面的0000 0000都放在ffc2~ffc4中,最后面的八位0000 1010放在ffc1中。

4.查看变量的内存地址

在调试过程中,我们经常会采取打印的方式查看变量的地址

1 #include <stdio.h>
2
3 int main()
4 {
5     int a = 10;
6     printf("变量a的地址是:%p", &a);
7     return 0;
8 }

第6行中的&是一个地址运算符,&a表示取得变量a的地址。格式符%p是专门用来输出地址的。输出结果是:

变量a的地址是:0x7fff5fbff8f8

这个0x7fff5fbff8f8就是变量a的内存地址

回到顶部

三、负数的二进制形式

1 int main()
2 {
3     int b = -10;
4     return 0;
5 } 

在第3行定义了一个整型变量,它的值是-10。-10在内存中怎样存储的呢?其实任何数值在内存中都是以补码的形式存储的。

  • 正数的补码与原码相同。比如9的原码和补码都是1001
  • 负数的补码等于它正数的原码取反后再+1。(取反的意思就是0变1、1变0)

那么-10的补码计算过程如下:

1> 先算出10的二进制形式:0000 0000 0000 0000 0000 0000 0000 1010

2> 对10的二进制进行取反:1111 1111 1111 1111 1111 1111 1111 0101

3> 对取反后的结果+1:1111 1111 1111 1111 1111 1111 1111 0110

因此,整数-10在内存中的二进制形式是:1111 1111 1111 1111 1111 1111 1111 0110

回到顶部

四、变量的作用域

1.作用域简介

变量的作用域就是指变量的作用范围。先来看看下面的程序:

1 int main()
2 {
3     int a = 7;
4
5     return 0;
6 }

  • 在第3行定义了一个变量a,当执行到这行代码时,系统就会为变量a分配存储空间
  • 当main函数执行完毕,也就是执行完第5行代码了,变量a所占用的内存就会被系统自动回收
  • 因此,变量a的作用范围是从定义它的那行开始,一直到它所在的大括号{}结束,也就是第3~6行,一旦离开这个范围,变量a就失效了

2.代码块

1> 代码块其实就是用大括号{}括住的一块代码。

 1 int main()
 2 {
 3     {
 4         int a = 10;
 5
 6         printf("a=%d", a);
 7     }
 8
 9     a = 9;
10
11     return 0;
12 }

  • 注意第3~7行的大括号,这就是一个代码块
  • 当执行到第4行时,系统会分配内存给变量a
  • 当代码块执行完毕,也就是执行完第6行代码后,变量a所占用的内存就会被系统回收
  • 因此,变量a的作用范围是从定义它的那行开始,一直到它所在的大括号{}结束,也就是第4~7行,离开这个范围,变量a就失效了
  • 所以,上面的程序是编译失败的,第9行代码是错误的,变量a在第7行的时候已经失效了,不可能在第9行使用

2> 如果是下面这种情况

 1 int main()
 2 {
 3     int a = 9;
 4
 5     {
 6         int a = 10;
 7
 8         printf("a=%d", a);
 9     }
10
11     return 0;
12 }

  • 注意第3、6行,各自定义了一个变量a,这种情况是没问题的。C语言规定:在不同作用域中允许有同名变量,系统会为它们分配不同的存储空间。
  • 在第3行定义的变量a的作用域是:第3~12行;在第6行定义的变量a的作用域是:第6~9行。
  • 最后注意第8行:尝试输出变量a的值。那这里输出的是哪一个变量a呢?先看输出结果:
a=10

这里采取的是“就近原则”,也就是第8行访问的是在第6行定义的变量a,并不是在第3行的变量a。

回到顶部

五、变量的初始化

变量在没有进行初始化之前,不要拿来使用,因为它里面存储的是一些垃圾数据

1 #include <stdio.h>
2
3 int main()
4 {
5     int c;
6
7     printf("%d", c);
8     return 0;
9 }

注意第5行的变量c,只是定义了变量,并没有给它赋初值。输出结果:

1606422622

可以发现,变量c里面存储的是一些乱七八糟的数据

时间: 2024-10-06 12:13:28

c语言章节6的相关文章

c语言章节5

本文目录 一.取值范围 二.char 三.说明符 四.自动类型提升 五.强制类型转换 C语言有丰富的数据类型,因此它很适合用来编写数据库,如DB2.Oracle等大型数据库都是C语言写的.其中,提供了4种最常用的基本数据类型:char.int.float.double,使用这些数据类型,我们就可以定义相应的变量来存储数据.这讲就来深入研究一下基本数据类型的一些使用细节. 回到顶部 一.取值范围 我们已经知道,不同数据类型所占的存储空间是不一样的.比如在64bit编译器环境下,char类型占用1个

c语言章节2

本文目录 一.基本概念 二.函数的定义 三.形式参数和实际参数 四.返回值类型 五.return 六.函数定义的注意 七.常见函数 前面已经讲完了C语言中的基本语句和基本运算了,这讲呢,介绍C语言中的重头戏---函数.其实函数这个概念,在大部分高级语言中都是非常重要的,我也已经在<第一个C语言程序>一讲中对函数作了一个简单介绍. 回到顶部 一.基本概念 1.什么是函数 任何一个C语言程序都是由一个或者多个程序段(小程序)构成的,每个程序段都有自己的功能,我们一般称这些程序段为“函数”.所以,你

c语言章节3

本文目录 前言 一.顺序结构 二.选择结构1-if语句 三.选择结构2-switch语句 四.循环结构1-while循环 五.循环结构2-do while循环 六.循环结构3-for循环 七.break和continue 回到顶部 前言 1.默认的运行流程 默认情况下,程序的运行流程是这样的:运行程序后,系统会按书写顺序执行程序中的每一行代码.比如下面的程序 1 #include <stdio.h> 2 3 int main() 4 { 5 6 printf("Hello-1\n&q

c语言章节13

本文目录 一.程序设计语言 二.是否需要计算机专业知识 三.是否要英文很好 四.开发工具 五.开发环境 六.Mac OS X的获取途径 七.调试设备 八.开发者证书 九.总结 在上一讲中,介绍了什么是iOS开发.说简单一点,iOS开发,就是开发运行在iPhone或者iPad上的软件.这么一说完,应该有很多人就会产生一些疑惑,比如学习iOS开发是不是一定要买iPhone?需不需要买苹果电脑呢?学软件开发是不是一定要计算机专业的?因此,在这讲中,我会为广大有意学习iOS开发的朋友们进行解惑,说明一下

c语言章节4

本文目录 一.算术运算符 二.赋值运算符 三.自增运算符和自减运算符 四.sizeof 五.逗号运算符 六.关系运算符 七.逻辑运算符 八.三目运算符 九.位运算符 计算机的基本能力就是计算,所以一门程序设计语言的计算能力是非常重要的.C语言之所以无所不能,是因为它不仅有丰富的数据类型,还有强大的计算能力.C语言一共有34种运算符,包括了常见的加减乘除运算.这讲就对C语言中的运算符做一个详细介绍. 回到顶部 一.算术运算符 算术运算符非常地简单,就是小学数学里面的一些加减乘除操作.不过呢,还是有

c语言章节11

本文目录 一.计算机常识 二.程序设计语言发展史 三.C语言简史 四.C语言的特点 五.C语言的作用 六.C语言的版本问题 前面已经给大家介绍了iOS开发相关的一些基础知识,比如学习iOS开发需要什么准备.iOS开发的前景等等.在<开篇>这讲中说过:其实iOS开发就是开发iPhone\iPad上的软件,而要想开发一款软件,首先要学习程序设计语言.iOS开发需要学习的主要程序设计语言有:C语言.C++.Objective-C,其中C++.Objective-C都是以C语言为基础,从C语言衍生出来

c语言章节9

本文目录 一.关键字 二.标识符 三.注释 上一讲中已经创建了第一个C语言程序,知道了C程序是由函数构成的,这讲继续学习C语言的一些基本语法.C语言属于一门高级语言,其实,所有的高级语言的基本语法组成部分都是一样的,只是表现形式不太一样.就好像亚洲人和非洲人,大家都有人类的结构:2只手.2只脚.1个头,只是他们外表不太一样,比如肤色.脸型.因此,你掌握好了一门高级语言,再去学习其他高级语言,那是相当快的.而且,很多其他高级语言,比如后面要学习的Objective-C,都是基于C语言.从C语言衍生

c语言章节8

本文目录 一.数据的存储 二.数据类型 三.常量 三.变量 在我们使用计算机的过程中,会接触到各种各样的数据,有文档数据.图片数据.视频数据,还有聊QQ时产生的文字数据.用迅雷下载的文件数据等.这讲我们就来介绍C语言中数据的处理. 回到顶部 一.数据的存储 1.数据类型 首先来看看计算机是怎么存储数据的.总的来说,计算机中存储的数据可以分为两种:静态数据和动态数据. 1> 静态数据 概念:静态数据是指一些永久性的数据,一般存储在硬盘中.硬盘的存储空间一般都比较大,现在普通计算机的硬盘都有500G

c语言章节14

本文目录 一.什么是iOS 二.主流手机操作系统 三.什么是iOS开发 四.学习iOS开发的目的 五.学习iOS开发的前提 从今天开始,我就开始更新[零基础学习iOS开发]这个专题.不管你是否涉足过IT领域,也不管你是理科生还是文科生,只要你对iOS开发感兴趣,都可以来阅读此专题.我尽量以通俗易懂的语言,让每个人都能够看懂.若遇到不明白的地方或者对此文有异议,望及时评论. 回到顶部 一.什么是iOS 要想学习iOS开发,首先要搞清楚什么是iOS.iOS其实是一款操作系统,就像平时我们在电脑上用的