【C语言】指针(一)

目录

一、内存

1.1 ❥ 理解内存和地址的关系

1.2 ❥ 编址

二、指针变量

2.1 ❥ 取地址操作符(&)

2.2 ❥ 指针变量和解引用操作符(*)

2.3 ❥ 指针变量的大小

三、指针类型的意义

3.1 ❥ 指针的解引用

3.2 ❥ 指针+-整数

3.3 ❥ 易错点

3.4 ❥ void*指针

四、野指针

4.1 ❥ 野指针的成因

4.2 ❥ 野指针可能导致的问题

4.3 ❥ 如何避免野指针的出现

五、指针运算

5.1 ❥ 指针 +- 整数

5.2 ❥ 指针 - 指针

5.3 ❥指针的关系运算(比大小)

六、二级指针


一、内存

内存是电脑上特别重要的存储器,计算机中程序的运行都是在内存中完成的。

所以为了有效的使用内存,就把内存分成一个个小的内存单元,每个内存单元的大小是1个字节(1 byte)

为了能够有效的访问到内存的每个单元,就给内存单元进行了编号,这些编号被称为该内存单元的地址。

1.1 ❥ 理解内存和地址的关系

我们可以举个例子:当我们回宿舍的时候,我们是如何找到自己所在的宿舍呢?

当然是根据门牌号来确定我们所在的宿舍。有了门牌号,就能提高效率,快速的找到房间。

一楼:101,102,103...
二楼:201,202,203...

这里的一个学生宿舍相当于一个内存单元,每个内存单元的大小取1个字节,1个字节里放8个比特位,就好比同学住的八人间,每个人是1个比特位。

每个内存单元都有一个编号,这个编号就相当于宿舍的门牌号,有了这个内存单元的编号,cpu就可以快速找到一个内存空间。

这里的门牌号,也就是内存单元的编号,也叫作地址。c语言中给地址起了个新名字,叫指针。

所以:内存单元的编号 == 地址 == 指针

1.2 ❥ 编址

CPU访问内存中的某个字节空间,必须知道这个字节空间在内存的什么位置,因为内存中字节很多,所以需要给内存进行编址。

计算机中的编址,并不是把每个字节的地址记录下来,而是通过硬件设计完成的。

计算机内是有很多的硬件单元,而硬件单元是要互相协同工作的。所谓的协同,至少相互之间要能够进行数据传递。

但是硬件与硬件之间是互相独立的,要想达到通信效果,那就得用线连接起来。
而CPU和内存之间也是有大量的数据交互的,所以,两者必须也用线连起来。其中,有一组线叫
地址总线。

32位机器有32根地址总线,每根线只有两态,表示0或者1(电脉冲有无),那么一根线,就能表示2种含义,2根线就能表示4种含义,依次类推。32根地址线,就能表示2^32种含义,每一种含义都代表一个地址。地址信息被下达给内存,在内存上,就可以找到该地址对应的数据,将数据在通过数据总线传入CPU内寄存器。



二、指针变量

2.1 ❥ 取地址操作符(&)

在c语言中,创建变量其实就是向内存申请空间:

上述的代码就是创建了整型变量a,内存申请了4个字节,用于存放整数10,其中每个字节都有地址,上图中4个字节的地址分别是:

0x010FF9E4

0x010FF9E5

0x010FF9E6

0x010FF9E7

这里我们利用取地址操作符(&)来获取a的地址。

#include <stdio.h>
int main()
{int a = 10;printf("%p\n", &a);//打印的是首字节的地址,即:0x010FF9E4return 0;
}

2.2 ❥ 指针变量和解引用操作符(*)

❤ 指针变量

那我们通过取地址操作符(&)拿到的地址是一个数值,比如:0x010FF9E4,这个数值有时候也是需要存储起来,方便后期再使用。那么我们把这样的地址值存放在指针变量。

int* p = &a;//将a的地址存放在指针变量p中
  1. p:指针变量,用来存放地址
  2. 存储地址的意义:p有机会找到a
  3. int:说明p指向的对象是int类型的。
  4. *:放在这里只是说明p是个指针变量(没有其它含义)
  5. int*:说明指针变量p的类型是int*

解引用操作符(*)

c语言中,我们可以通过解引用操作符来找到我们指针(地址)所指向的对象。

代码如下:

这里*p的意思是:通过p中存放的地址,找到p所指向的对象,*p就是p所指向的对象a

当然,也可以对*p(即变量a)进行修改,代码如下:

2.3 ❥ 指针变量的大小

  • 不管什么类型的指针(地址),都是在创建指针变量,指针变量是用来存放地址的。
  • 指针变量的大小取决于一个地址存放需要多大的空间(即指针变量的大小取决于地址的大小)。
  • 32位机器上的地址:32bit位 - 4byte ,所以指针变量的大小是4个字节
  • 64位机器上的地址:64bit位 - 8byte ,所以指针变量的大小是8个字节
  • 指针变量的大小和类型是无关的,只要指针类型的变量,在相同的平台下,大小都是相同的。
  • 在32OS下,int*,char*,double*...都是4个字节。

总结:指针大小与数据类型无关,它与系统位数有关。

三、指针类型的意义

指针变量的大小和类型无关,只要是指针变量,在同一平台下,大小都是一样的,为什么还要有各种指针类型呢?

其实指针类型是有它存在的意义的,我们来看下指针类型存在哪些意义吧。

3.1 ❥ 指针的解引用

我们首先来看下面两段代码:对他们进行调试:

代码1

代码2

调试我们可以看到:代码1会将n的4个字节全部改为0,但是代码2只是将n的第一个字节改为0。

结论:指针的类型决定了,指针在被解引用时能有多大权限(一次能操作几个字节)。

例如:char*的指针解引用就只能访问1个字节,而int*的指针解引用能访问4个字节。

3.2 ❥ 指针+-整数

我们还是先来看一段代码:

#include <stdio.h>
int main()
{int a = 10;char* pa = (char*)&a;int* pc = &a;printf("%p\n", &a);printf("%p\n", pa);printf("%p\n", pa + 1);printf("%p\n", pc);printf("%p\n", pc + 1);return 0;
}

运行结果可以发现:char* 类型的指针变量+1跳过1个字节;int* 类型的指针变量+1跳过了4个字节。

结论:指针的类型决定了指针在+-操作的时候,能够跳过几个字节。(决定了指针的步长)

3.3 ❥ 易错点

我们看如下代码:

#include <stdio.h>
int main()
{int a = 0;int* pa = &a; float* pf = &a; 		return 0;

发现:

pa解引用访问4个字节,pa+1跳过4个字节。

pf解引用访问4个字节,pf+1也是跳过4个字节。

那么int*和float*是不是就可以通用呢?

答案是:不能。

原因:因为两者的存储方式完全不同。

3.4 ❥ void*指针

void* —— 泛型指针,没有具体类型的指针。

这种类型的指针可以用来接收任意类型的地址。

如下面代码所示:

#include <stdio.h>
int main()
{int a = 10;void* pa = &a;void* pc = &a;return 0;
}

但是也有局限性:void*类型的指针不能直接解引用和进行指针的 + - 整数的运算,因为不知道访问(多大空间)多少个字节。

如下图所示:

void*类型的指针的用处:

一般void*类型的指针是使用在函数参数的部分,用来接收不同类型数据的地址,这样涉及可以实现泛型编程的效果。使得一个函数来处理多种类型的数据。

四、野指针

  • 当一个指针没有被正确初始化或者被错误释放后,它可能仍然保留着之前所指向的内存地址。在这种情况下,这个指针就成了野指针。
  • 它包含一个无效的内存地址。当野指针被解引用或者试图在其指向的内存上进行读取或写入操作时,就会导致未定义行为。

4.1 ❥ 野指针的成因

 

❤  指针没有初始化

一个指针变量被声明但没有初始化时,其中的值将是不确定的,它可能指向任意的内存地址,包括重要的内存位置。

❤ 指针越界访问

❤ 指针指向的空间释放

当某个内存块已经被释放并返回给系统,但对应的指针仍然保留着指向之前的内存地址,那么这个指针就是野指针。

分为以下几个方面:

  1. 一个指针指向已经释放的内存区域(例如通过free函数释放的内存)。
  2. 一个指针指向已经超出其作用域的静态变量,这可能会访问到无效的内存。
  3. 一个指针指向栈帧上的局部变量,并且该变量的生命周期已经结束(例如函数调用返回后),这时使用该指针会访问到无效的内存。

例如下面代码:

#include <stdio.h>
int* test()
{int n = 100;//局部变量n进栈创建,出栈销毁return &n;
}int main()
{int* p = test();printf("%d\n", *p);return 0;
}
  • 空间销毁的意思是:内存里这块空间还存在,只是不属于我当前程序了。
  • 这块空间是我当前程序所拥有的,我就可以使用它,这个函数出去之后,这块空间就还给操作系统了,我当前程序没有这块空间使用权限了。但是内存里的这块空间还存在。

4.2 ❥ 野指针可能导致的问题

  1. 未定义行为:使用野指针进行解引用或间接访问可能会导致未定义行为。这包括程序崩溃,数据损坏或产生不可预测的结果。
  2. 安全漏洞:如果野指针指向重要的内存区域,可能会导致安全漏洞。攻击者可能利用野指针来修改或获取敏感数据,从而对程序的安全性产生影响。
  3. 内存泄漏:如果野指针指向已经释放的内存,可能导致内存泄漏。释放的内存应该被正确管理,否则会占用系统资源并影响程序性能。

4.3 ❥ 如何避免野指针的出现

  • ❤ 指针初始化
  • ❤ 小心指针越界
  • ❤ 指针指向的空间释放要及时置空(NULL),或者不想使用的时候赋值空指针(NULL),或者一个指针不想初始化的时候可赋值空指针(NULL)
  • ❤ 避免返回局部变量的地址
  • ❤ 谨慎使用动态内存分配和释放,确保指在针使用之前检查其有效性

补充:NULL是C语言中定义的一个标识符常量,值是0,0也是地址。该地址是属于系统内核的,用户程序是不能使用的,读写该地址会报错。

五、指针运算

5.1 ❥ 指针 +- 整数

因为数组在内存中是连续存放的,只要知道第一个元素的地址,就能找到后面的所有元素。

#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int* p = &arr[0];int i = 0;int sz = sizeof(arr) / sizeof(arr[0]);for (i = 0; i < sz; i++){printf("%d ", *(p + i)); //p+i 这里就是指针+整数}return 0;
}

5.2 ❥ 指针 - 指针

指针 - 指针的绝对值得到的是指针和指针之间的元素个数

注意:

  1. 不是所有的指针都能相减,指向同一块空间的两个指针才能相减,这样才有意义。
  2. 指针(地址)+ 指针毫无意义。

5.3 ❥指针的关系运算(比大小)

标准规定:允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但不允许与指向第一个元素之前的那个内存位置的指针进行比较。

代码1

代码2

实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。

六、二级指针

二级指针:用来存放指针变量的地址的指针(变量)

如下图所示:

对于二级指针的运算有:
 *ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是pa 

int b = 20;
*ppa = &b; //等价于 pa = &b;

**ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作: *pa ,找到的是 a 

**ppa = 30;
//等价于*pa = 30;
//等价于a = 30;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/695843.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何将 DFMini player MP3 模块与 Arduino 结合使用

要创建此项目&#xff0c;您将使用&#xff1a; DFPlayer迷你MP3模块 10kΩ电阻 开关按钮 面包板 Arduino UNO 杜邦线 现在&#xff0c;我们将学习如何构建该项目。 什么是DF Mini Player MP3模块 DFMini Player 模块是一个小型音乐播放器。它成本低、功耗低&#xff0c;可…

JSP ssm 房屋中介管理myeclipse开发mysql数据库springMVC模式java编程计算机网页设计

一、源码特点 JSP ssm 房屋中介管理系统是一套完善的web设计系统&#xff08;系统采用SSM框架进行设计开发&#xff0c;springspringMVCmybatis&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采…

woffice– 内部网和外部网WordPress高端资讯主题

下载地址&#xff1a;https://m.gx.cn/site/3046.html 完全灵活&#xff0c;与最新的WordPress品牌兼容 翻译语言超过15种 使用最新技术设计快速web应用程序 所有这些都以谷歌材料设计为灵感&#xff0c;采用易于定制的设计&#xff0c;给人一种优美的现代感和易于导航的用户…

企业为什么需要HTTPS

一.什么是HTTPS HTTPS &#xff08;全称&#xff1a;Hyper Text Transfer Protocol over SecureSocket Layer&#xff09;&#xff0c;是以安全为目标的 HTTP 通道&#xff0c;在HTTP的基础上通过传输加密和身份认证保证了传输过程的安全性 。HTTPS 在HTTP 的基础下加入SSL&a…

01-02-2

1、typedef的使用 a.语法 typedef 原名 别名&#xff1b;。 ​ typedef struct student {int num;char name[20];char sex; }stu,*pstu;//stu相当于struct student这个类型&#xff0c;*pstu相当于struct student * 别名的理解方法&#xff1a;若是字母前面有符号&#xff0…

安全设备篇——日志审计系统

日志审计是企业网络安全建设中重要的一环。 其实日志审计在结构上和部分原理上和态势感知很相近&#xff0c;都进行数据采集&#xff0c;都需要强调规则&#xff0c;老规矩先放百度解释&#xff1a; “日志审计系统是用于全面收集企业IT系统中常见的安全设备、网络设备、数据库…

微信小程序的设计与实现

微信小程序的设计与实现 目录 1.系统简述&#xff1a; 2.开发工具及相关技术&#xff1a; 2.1 HTML、WXSS、JAVASCRIPT技术 2.2 Vanilla框架 2.3 uni-app框架 2.4 MYSQL数据库 3.工程结构及其说明&#xff1a; 4.主要功能展示 4.1登录 4.2 注册 4.3 首页…

怎么把图片大小缩小到1M?教你几招图片你压缩

当我们的图片数量越来越多的时候&#xff0c;占用的内存也就越来越多&#xff0c;时间长了之后&#xff0c;会导致我们空间不足或者设备比较卡顿&#xff0c;为了缓解这个问题&#xff0c;很多人会选择去删除一些不必要的图片文件&#xff0c;其实还有个方法就是利用图片压缩的…

视频号小店怎么做?五个步骤教会你,快来学习吧!

大家好&#xff0c;我是电商糖果 今年关于视频号小店真的很火&#xff0c;不少朋友都找糖果咨询过关于做店的方法。 糖果做小店已经快两年了&#xff0c;对小店运营方面可以说比较了解。 这里就把小店的整体做店流程分为五步&#xff0c;一步步讲解给大家。 第一步&#xff…

js图片回显的方法

直接上代码&#xff1a; <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title></head><body>// HTML部分<input type"file" id"fileInput"><button onclick"show…

【重装系统】U盘制作软件——Rufus

Rufus 是一款格式化和创建 USB 启动盘的辅助工具。 本软件适用于以下场景&#xff1a; 需要将可引导 ISO (Windows、Linux、UEFI 等) 刻录到 USB 安装媒介的情况 需要处理未安装操作系统的设备的情况 需要在 DOS 环境下刷写 BIOS 或其他固件的情况 需要运行低级工具的情况 官…

二叉树之左叶子的和

题目&#xff1a; 给定二叉树的根节点 root &#xff0c;返回所有左叶子之和。 示例 1&#xff1a; 输入: root [3,9,20,null,null,15,7] 输出: 24 解释: 在这个二叉树中&#xff0c;有两个左叶子&#xff0c;分别是 9 和 15&#xff0c;所以返回 24示例 2: 输入: root […