第一、了解内存空间

本文章文字有点多,会有点枯燥,配合图文一起看可以缓解枯燥,耐心阅读哦!!!

先了解内存地址,才更好的理解指针!

我们可以把内存想象为成一列很长很长的货运火车,有很多大小相同的车厢,而每个车厢正好相当于在内存中表示一个字节。这些车厢装着不同的货物,就像我们的内存要存着各式各样的数据。

多啰嗦一下

我们平时在电脑上能够听音乐、看视频和文章,其实看到的这些东西就是内存中每个“车厢”里面的数据,这些数据最终还是由二进制0/1演变而成。

虽然视频、文章、音乐等这些信息在我们眼里是不同的,但对于计算机来说它们在内存中都是以二进制的形式来表示。

因为我们要知道去哪存或取数据,所以内存中每个字节都有对应的编号,就像火车上的车厢编号一样。而这个内存中每个字节的编号就是我们常说的内存地址,是按一个字节接着一个字节的次序进行编址。如下图所示:

凡事多问几个为什么?

1. 为什么内存地址都有0x开头?

0x 开头代表以十六进制来表示的意思。

2. 为什么我们平时看到内存地址是这样的呢?如图:

因为内存容量很大,容量大字节数自然也多了,所以需要更多位来编址内存地址。上图的(0x00 ...)内存地址这里只是便于理解!

3. 为什么我那么菜呢?

哈哈哈......你心里没点*数吗?

关于内存字节

  • 1个内存地址只存1个字节 (Byte);
  • 1个字节等于8位二进制,每一位二进制的0或1,叫“比特”(bit);
  • 比特是最小单位,字节是比特的集合,也是一个单位;

内存给数据类型地址分配如下:

  • char:占一个字节分配一个地址;
  • int: 占四个字节分配四个地址;
  • 还有long、float、double等类型,等着你来动手测试。

可以使用sizeof进行验证:

#include<stdio.h>
int main () {
printf("sizeof(char)=%u\n",sizeof(char));
printf("sizeof(int)=%u\n",sizeof(int));
return ;
}

结果如下:

第二、理解指针

不要把指针想得太复杂,指针的实质就是内存“地址”,可以说指针就是地址,其实指针就是保存地址的变量。

拿普通变量跟指针变量做比较:

char a;     // 定义一个变量a,用于保存char类型的数据;
char *b; // 定义一个指针变量b,用于保存一个内存地址,这个内存地址上的数据必须是char类型的。

举个例子,给指针变量进行赋值:

#include<stdio.h>
int main () {
char a = ; // char 类型占一个字节;
char *b = &a; // “&”是取变量的地址,取出a在内存中的地址;
// 赋值给b指针,此时b变量存储的就是a地址。
printf("我是a变量的值:%d\n",*b); // *b表示输出b里面存储的地址上的数据;
// 证明b上存储的是a的地址;
printf("我是a的地址:%p\n",&a);
printf("我是b变量的值:%p\n",b);
return ;
}

输出结果为:

我是a变量的值:
我是a的地址:000000000062FE17
我是b变量的值:000000000062FE17

通过画图来理解:

通过指针间接性修改变量的值

char a = ;
char *b = &a;
printf("初始值:a=%d,*b=%d\n",a,*b);
*b = ; // 其实操作的就是变量a本身的值;
printf("修改后:a=%d,*b=%d\n",a,*b);
------------------------------------------
输出结果为:
初始值:a=,*b=
修改后:a=,*b=

指针类型的概念

我们知道char类型的数据只占一个字节,有很多类型是需要多个字节来存储的,像int类型的数据就需要四个字节来存储(根据平台不同,长度也有可能不一致)。

对于int类型的指针从当前字节(地址)开始共四个字节(地址)都是属于该变量的值, 而对于char类型则只表示当前字节(地址)。代码如下:

int a = ;
int * p1 = &a;
char * p2 = (char *)&a; // 这里需要强制转换一下类型
printf("*p1=%d,*p2=%d\n",*p1,*p2);
-----------------------
输出:*p1=,*p2=

通过画图来便于理解:

通过上文我们已经对int类型指针有所了解了,*p1的输出是在我们预算范围之内的,但是为什么*p2输出的值是3呢?

重点,敲黑板!!!

因为计算机是使用二进制来表示数字的,上面(259)十进制转换二进制是 [100000011],由于一个int类型变量占用四个字节,8位二进制为一个字节,补齐高位的0后,则 [00000000 00000000 00000001 00000011],每8位二进制(一个字节)换算为十进制,则 [0  0  1  3]。

此时你应该差不多明白*p2为什么输出的值为3了吧,但是内存地址中有个概念叫"大小端模式",就会有两种不同的排序:[0  0  1  3] or [3  1  0  0]。

由于计算机读取*p2的地址是0x00,所以直接输出这个地址上的数据,你也可以试着改一下,把259换成258/257等,看看是否正如所说。

验证它们存储地址,代码如下:

int a = ;
int * p1 = &a;
char * p2 = (char *)&a;
printf("*p1=%d,*p2=%d\n",*p1,*p2);
printf("&a=0x%p\n",&a);
printf("p1=0x%p\n",p1);
printf("p2=0x%p\n",p2);

输出结果正如我们预想的:

当你看到这里的时候,你只是刚刚认识指针而已,以上是我们俗称的一级指针,一级指针是比较简单的,还有二级指针和多级指针,更绕、更难理解,接下来介绍一下二级指针。

在讲二级指针前,我们是否有疑问:什么是一级指针?什么是二级指针呢?两者有什么区别呢?

  • 一级指针存储变量的地址,通过这个地址"直接获取"变量的数据。
  • 二级指针存储一级指针的地址,二级指针通过一级指针"间接获取"获取变量的数据。
  • 多级指针以此类推,个人理解,讲的不对欢迎指正。

再坚持一下,精彩在"下面"!!![/滑稽]

二级指针

“指针的指针”也就是我们俗称的二级指针。

什么是“指针的指针”,例如下面代码:

char a = ;
char * p1 = &a;
char ** p2= &p1;
printf("*p=%d,**p2=%d\n",*p1,**p2); // 输出:*p1=5,**p2=5

通过画图来理解:

多级指针也就是指针的指针的指针.....,以此类推即可。

第三、指针运算问题

指针运算是根据指针的类型不同而进行运算的,因类型的不同,在加1/减1操作时,内存分配的空间也不同。

又拿int类型和char类型来作比较,代码如下:

char类型+1:从输出结果可以看出地址是递增1的,正符合char类型占一个字节的说法。

char c = 'h';
char *a = &c;
for (int i=;i<;i++){
printf("a+1=0x%p\n",a + i);
}
--------------------------------
输出结果:
a+=0x000000000062FE0F
a+=0x000000000062FE10
a+=0x000000000062FE11

int类型+1:输出的地址之间相差为4,正是int类型占据空间。

int c = ;
int *a = &c;
for (int i=;i<;i++){
printf("a+1=0x%p\n",a + i);
}
--------------------------------
输出结果:
a+=0x000000000062FE0C
a+=0x000000000062FE10
a+=0x000000000062FE14

char类型和int类型分别+1在内存中地址分配,如图:

指针就介绍到这里,这只是指针的基础,还有数组指针、指针数组、null指针、void指针等等知识,还需要学习,后续继续更新。

以上有不恰当或者讲得不对的地方,希望各位留言指正,谢谢!

站在巨人的肩膀上!

C语言指针及占据内存空间的更多相关文章

  1. C语言中指针占据内存空间问题

    以前一直有个疑问,指向不同类型的指针到底占用的内存空间是多大呢? 这个问题我多次问过老师,老师的答案是"指向不同类型的指针占据的内存空间大小不同",我一直很之一这个答案,今天我就做 ...

  2. 在64位系统下,指向int型的指针占的内存空间多大?

    不废话,请看代码演示如下: 注意使用的操作系统的位数,不同位数的操作系统,结果不一样! 我是用的是64位的操作系统! linux下示例代码如下: #include <stdio.h> in ...

  3. java内存空间详解

    Java内存分配与管理是Java的核心技术之一,之前我们曾介绍过Java的内存管理与内存泄露以及Java垃圾回收方面的知识,今天我们再次深入Java核心,详细介绍一下Java在内存分配方面的知识.一般 ...

  4. 不可或缺 Windows Native (7) - C 语言: 指针

    [源码下载] 不可或缺 Windows Native (7) - C 语言: 指针 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 指针 示例cPointer.h #i ...

  5. java内存空间

    Java内存分配与管理是Java的核心技术之一,之前我们曾介绍过Java的内存管理与内存泄露以及Java垃圾回收方面的知识,今天我们再次深入Java核心,详细介绍一下Java在内存分配方面的知识.一般 ...

  6. 深入理解C语言 - 指针使用的常见错误

    在C语言中,指针的重要性不言而喻,但在很多时候指针又被认为是一把双刃剑.一方面,指针是构建数据结构和操作内存的精确而高效的工具.另一方面,它们又很容易误用,从而产生不可预知的软件bug.下面总结一下指 ...

  7. C++内存空间管理

    C++内存空间管理 1.C++内存机制 1.栈(Stack),函数中的局部变量,由编译器负责分配释放,函数结束,变量释放. 2.堆(Heap),通过new 申请的内存,由delete或delete[] ...

  8. 嵌入式C语言4.1 C语言内存空间的使用-指针

    指针:就是内存资源的地址.门牌号的代名词 假如你所在的城市是一个内存(存储器),如果找到你家,就是通过你的家庭住址(指针)寻找,而你家里的摆设面积之类的就是内存的内容(指针指向的内容). 指针变量:存 ...

  9. C语言重点——指针篇(一文让你完全搞懂指针)| 从内存理解指针 | 指针完全解析

    有干货.更有故事,微信搜索[编程指北]关注这个不一样的程序员,等你来撩~ 注:这篇文章好好看完一定会让你掌握好指针的本质 C语言最核心的知识就是指针,所以,这一篇的文章主题是「指针与内存模型」 说到指 ...

随机推荐

  1. 深度兴趣网络DIN-SIEN-DSIN

    看看阿里如何在淘宝做推荐,实现"一人千物千面"的用户多样化兴趣推荐,首先总结下DIN.DIEN.DSIN: 传统深度学习在推荐就是稀疏到embedding编码,变成稠密向量,喂给N ...

  2. CEF编译遇到的问题记录

    在使用vs2015编译cef官方代码的时候遇到很奇怪的问题, 我用官方的demo cefsimple例子程序编译debug版本 可以正常编译打开网页 正常的打开 我把官方的例子单独创建一个新的项目编译 ...

  3. Redo与Undo的理解

    本文概要本文的原意是一篇个人学习笔记,为了避免成为草草记录一下的流水账,尝试从给人介绍的角度开写.但在整理的过程中,越来越感觉力不从心,一是细节太多了,原以为足够了解的一个小知识点下可能隐藏了很多细节 ...

  4. 1042 字符统计 (20 分)C语言

    请编写程序,找出一段给定文字中出现最频繁的那个英文字母. 输入格式: 输入在一行中给出一个长度不超过 1000 的字符串.字符串由 ASCII 码表中任意可见字符及空格组成,至少包含 1 个英文字母, ...

  5. 1041 考试座位号 (15 分)C语言

    每个 PAT 考生在参加考试时都会被分配两个座位号,一个是试机座位,一个是考试座位.正常情况下,考生在入场时先得到试机座位号码,入座进入试机状态后,系统会显示该考生的考试座位号码,考试时考生需要换到考 ...

  6. spring boot集成spring-boot-starter-mail邮件功能

    前情提要 以目前IT系统功能来看,邮件功能是非常重要的一个功能.例如:找回密码.邮箱验证,邮件动态码.忘记密码,邮件营销等,都需要用到邮件功能.结合当下最流行的spring boot微服务,推出了sp ...

  7. bash的默认组合键

    组合键 组合按键 执行结果 Ctrl+C 终止目前的命令 Ctrl+D 输入结束(EOF),例如邮件结束的时候 Ctrl+M 就是Enter啦! Ctrl+S 暂停屏幕输出 Ctrl+Q 恢复屏幕输出 ...

  8. Date类与日期格式

    Date类概述: 表示特定的瞬间,精确到毫秒. Date()分配 Date 对象并初始化此对象,以表示分配它的时间(精确到毫秒).Date(long date)分配 Date 对象并初始化此对象,以表 ...

  9. 原生javascript实现选项卡(基础版)

    一.实现原理 1.主要运用“排他思想”,在设置当前元素前,先把相应元素恢复到默认状态 2.给相应元素添加下标的应用 二.代码展示 <!DOCTYPE html> <html> ...

  10. C++Primer第五版 6.1节练习

    练习6.1:实参和形参的区别是什么? 通俗解释: 实参是形参的初始值.编译器能以任意可行的顺序对实参求值.实参的类型必须与对应的形参类型匹配. 详解1) 形参变量只有在函数被调用时才会分配内存,调用结 ...