C语言-指针(一)

目录

指针

内存

概念

指针变量

取地址操作符(&)

操作符“ * ”

指针变量的大小

注意

指针类型的意义

作用

void * 指针

const修饰指针变量

const放在*前

const放在*后

双重const修饰

指针的运算

1.指针 + - 整数

2.指针 - 指针

3.指针的关系运算

野指针

概念

野指针形成原因

1.指针未初始化

2.指针越界访问

3.指针指向的空间释放

避免野指针的措施

assert断⾔

使用assert的好处

传值调用和传址调用

传值调用

传址调用

注意:


指针

内存

内存是计算机储存数据的地方

计算机常用内存单位

bit - ⽐特位          1byte = 8bit     满8比特进1
byte - 字节           1KB = 1024byte   满1024进1
KB     千字节         1KB = 1024byte
MB     兆字节         1GB = 1024MB
GB     吉字节         1TB = 1024GB
TB     太字节         1PB = 1024TB
PB     拍字节               

概念

把内存分为一个个内存单元,每个内存单元大小取1字节

每个内存单元都有一个编号,这个编号就是内存中的地址

计算机可以通过这个地址找到对应数据所在的地方,我们把这个地址取个新名字,指针

即 内存单元的编号 == 地址 == 指针

指针变量

用于存放指针的变量(即存放地址的变量)称为指针变量

结果为:类型  *   变量名

int b = 10;
int * a = &b
指针变量的类型由取地址的变量决定的
int类型变量取的地址,指针变量就是int
char类型取的地址,指针变量就是char

取地址操作符(&)

C语言使用 “ & ” 来获取变量在内存中的地址。

因为一个字节就有一个地址,而 int 具有四个字节,所以有四个地址

“ & ”在取地址的时候是取最小的地址即取第一个字节的地址

结构为 :&   (要取地址的变量名)

int a = 10;
int *ptr = &a; // ptr存储a的地址
printf("a的地址:%p\n", &a); // 输出a的内存地址

操作符“ * ”

该操作符一共有两个作用

1,声明指针变量

在变量声明时,“ * ” 用于表示该变量是一个指针,指向某种数据类型

2,解引用指针

表示获取指针指向的内存地址中的(即获取这个地址存放的值)

int main() {int a = 10;int* p = &a;*p = 0;            使用效果等于a = 0printf("%d", a);   *p获取了地址p的存放的值return 0;}

指针变量的大小

因为指针是用来存放地址的,所以他的大小是与存放的地址大小决定

在32位的计算机中,一个有32个地址总线,32个地址线产生的二进制作为地址,那么一个地址就有32个二进制,一个二进制需要一个bit存放,所以32位计算机的指针变量大小是4个字节

以此列推,64位计算机的指针变量就需要8个字节来存放

注意

指针的大小于类型无关,因此在同一个平台小所以的指针的大小都是相同的

计算机是使用二进制来存放地址的,但打印的地址是十六进制,是因为二进制的长度太长所以使用十六进制来表示,使用二进制的话一个地址就有32/64位了

指针类型的意义

指针不能使用类型来决定指针变量的大小,但不能把指针类型看成可有可无的东西

作用

1,指针类型决定了 解引用 进行操作时能够访问几个字节

2,当指针 +- 上整数(指针运算时)时,地址的位置会发生变化

void * 指针

表示无具体类型的指针(也叫泛型指针)

该类型指针可以用来接收任意类型的地址

但不能进行指针+-整数(指针的运算)和解引用的操作

const修饰指针变量

用于修饰指针变量时,表示该指针变量的值具有了常属性,不能被直接修改

虽然指针变量具有了常属性,但本质还是变量,不能当作常量来使用

但在C++中const修饰的变量就是常量,可以当作常量来使用(语言上的不同,于编译器无关)

const修饰指针时有三种方式:

const放在*前

int const * a = &b;

修饰指针指向的数据(数据不可变)

可以指向不同的地址,但不能修改指针指向的数据

const放在*后

int * const a = &b;

修饰指针指向的地址(地址不可变)

可以使用解引用来修改地址中的值,但不能修改指向的地址

口诀:左定数据,右定指针

双重const修饰

const int * const p;

const同时修饰地址和指针指向的数据

地址和数据都不能修改

指针的运算

指针变量存储的是内存地址,而指针运算(地址的运算)的本质是根据数据类型大小调整偏移量

指针的运算一共有三种方式:

1.指针 + - 整数

整数表示的是偏移的类型大小的个数,

例:类型为int,整数为2,指针+2时,此时所表示的意思是,地址向高地址移动两个int类型大小的字节个数,即向后移动四个字节,而指针 - 整数时,地址向低地址偏移

(高地址:大的地址,低地址:小的地址)

新地址 = 原地址 ± (整数 * sizeof(指针类型))

指针(地址)+ 整数(元素个数)=  新指针(新地址)

2.指针 - 指针

指针 - 指针的本质是两个地址相减

得出的整数是两个地址之间的元素个数

注意:

计算的前提是两个指针指向的是同一个空间

指针(地址) - 指针(地址)=  整数(元素个数)

3.指针的关系运算

指的是比较两个指针所指向的内存地址的大小关系

指针的关系运算符包括:==(相等)、!=(不等)、<(小于)、>(大于)、<=(小于等于)、>=(大于等于)

野指针

概念

野指针:指针指向的位置是未知的,无效的地址,用它们可能导致程序崩溃或未定义行为

野指针形成原因

1.指针未初始化

未初始化的指针可能指向随机内存地址

int *ptr; // 未初始化
*ptr = 100; // 可能覆盖任意内存区域(导致崩溃或数据损坏)

2.指针越界访问

指针访问了超出其合法分配范围的内存区域

通常发生在操作数组或动态分配的内存时,指针的偏移量超出了实际分配的空间,导致访问无效或受保护的内存地址

int arr[5] = {1, 2, 3, 4, 5};
int *p = arr;// 正确访问:p[0]到p[4]
for (int i = 0; i < 5; i++) {printf("%d ", p[i]);
}// 越界访问:p[5](超出数组范围)
printf("%d", p[5]); // 未定义行为!可能输出垃圾值或崩溃

3.指针指向的空间释放

指针指向的空间被释放后,若指针未被正确处理,就会变成“野指针”

int *ptr = malloc(sizeof(int)); // 分配内存,ptr指向合法地址(如0x1000)
*ptr = 42;                       // 合法操作:向0x1000写入数据
free(ptr);                       // 释放内存:0x1000被系统回收,但ptr的值仍为0x1000
// 此时ptr变为野指针!
*ptr = 100;                      // 危险操作:向已释放的地址0x1000写入数据(可能导致崩溃)

避免野指针的措施

1.注意指针的 初始化 和 地址范围 避免出现野指针

2.将不再使用 或 目前还不知道怎么使用的指针变量初始化为 NULL

NULL:是C语言的标识化常量,值是0,地址也是0,该地址不能使用,使用会导致报错

#include <stdio.h>
#include <stdlib.h>int main() {// 1. 动态分配内存int *ptr = malloc(sizeof(int));if (ptr == NULL) {printf("内存分配失败!\n");return 1;}// 2. 使用指针*ptr = 42;printf("存储的值: %d\n", *ptr);// 3. 释放内存,并立即置空指针free(ptr);ptr = NULL; // 关键步骤:标记指针为无效// 4. 后续操作(安全检测)if (ptr != NULL) {// 由于ptr已被置空,此代码块不会执行printf("尝试访问已释放的内存: %d\n", *ptr);} else {printf("指针已置空,无法访问。\n");}// 5. 尝试重复释放(安全)free(ptr); // free(NULL) 是安全的空操作printf("程序安全结束。\n");return 0;
}

3.避免返回局部变量的地址

局部变量在函数执行结束时会被销毁(栈内存回收),返回其地址会导致指针指向无效内存

本质上也是指针指向的空间被释放了

assert断⾔

assert断言是一个宏,定义在头文件 assert.h 中,用于在运行时确保程序满足指定的要求,如果条件不满足(即表达式为假),assert会终止程序并输出错误信息,帮助开发者快速定位代码中的逻辑错误或假设不成立的情况

assert的语法结构

#include <assert.h>  // 必须包含头文件
assert(condition);   // condition 是要检查的条件表达式#include <stdio.h>
#include <assert.h>  // 必须包含 assert.hint main() {int array[] = {10, 20, 30};int index = 3;  // 索引值(故意越界)// 断言:检查索引是否在合法范围内 [0, 2]assert(index >= 0 && index < 3);  // 条件失败时触发断言printf("array[%d] = %d\n", index, array[index]);return 0;
}运行结果Assertion failed: index >= 0 && index < 3, file example.c, line 8
Aborted (core dumped)

使用assert的好处

assert可以自动标识出错的文件和行号。

在不需要判断是否出错时,不需要更改代码就可以决定assert的开启和关闭

当不需要assert是就在头文件#include <assert.h >定义一个宏,前加上#include NDEBUG

#define NDEBUG
#include <assert.h >

传值调用和传址调用

传值调用 和 传址调用 是函数传递参数的两种基本方式,它们的核心区别在于是否直接操作原始数据

传值调用

函数接收的是参数的副本,而非原始数据本身

传值调用传递的是变量的值,只能使用传递过来的值进计算等操作

不能通过地址直接更改地址储存的值

传址调用

函数接收的是实参的地址(指针),通过地址直接操作原始数据

在函数里修改的值会直接影响该地址的原始数据,下次使用该地址的值时,调用的是修改后的值

特性传值调用传址调用
传递内容值的副本内存地址(指针)
是否影响实参不影响影响
内存占用形参和实参独立形参指向实参的内存
性能可能产生拷贝开销(大对象时)更高效(仅传递地址)
安全性原始数据安全需谨慎防止意外修改

注意:

数组名会隐式转换为指向首元素的指针(即数组名表示的是首元素的地址)

int *p = arr;  等价于&arr

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

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

相关文章

华为云Astro大屏连接器创建操作实例:抽取物联网iotda影子设备数据的连接器创建

目录 样图(API连接器创建成功) 说明 操作场景(以Astro大屏抽取iotda影子参数为例) 实际操作步骤 新建连接器 设置基本信息。 接口鉴权方式,支持API鉴权、AK/SK、API Key和无身份验证 无身份验证 AK/SK认证(目前暂不能用) API Key认证(第三方使用) API鉴权认…

【硬件系统架构】哈佛架构

一、引言 在计算机科学的浩瀚宇宙中&#xff0c;计算机体系结构犹如星辰般繁多且各有独特光芒。哈佛架构便是其中一颗耀眼的明星&#xff0c;它在众多计算机体系结构中占据着独特而重要的地位。从计算机技术的萌芽期一路走来&#xff0c;哈佛架构不断发展演变&#xff0c;在不同…

华为eNSP:IS-IS认证

一、什么是IS-IS认证&#xff1f; 华为eNSP中的IS-IS认证 IS-IS认证是华为eNSP网络中用于保障中间系统到中间系统&#xff08;IS-IS&#xff09;协议通信安全性的核心机制&#xff0c;通过身份验证和数据完整性校验防止非法路由信息注入或篡改。其实现方式与关键特性如下&…

如何创建并使用极狐GitLab 项目访问令牌?

极狐GitLab 是 GitLab 在中国的发行版&#xff0c;关于中文参考文档和资料有&#xff1a; 极狐GitLab 中文文档极狐GitLab 中文论坛极狐GitLab 官网 项目访问令牌 在极狐GitLab 16.1中引入添加了默认前缀。 项目访问令牌类似于密码&#xff0c;但你可以 限制访问资源&#xf…

C# 异步详解

C# 异步编程详解 一、异步编程基础概念 1. 同步 vs 异步 ​​同步(Synchronous)​​&#xff1a;任务按顺序执行&#xff0c;前一个任务完成后才会执行下一个​​异步(Asynchronous)​​&#xff1a;任务可以非阻塞地启动&#xff0c;主线程可以继续执行其他操作 2. 异步编…

C++ 之 【模拟实现 list(节点、迭代器、常见接口)】(将三个模板放在同一个命名空间就实现 list 啦)

1.前提准备 (1) list 的底层结构一般是带头双向循环链表 (1)为避免命名冲突&#xff0c;需要创建一个命名空间来存放模拟实现的 list (2)下面模拟实现list时&#xff0c;声明和定义不分离(具体原因后续讲解) 2.完整实现 2.1 链表节点 template<class T>//节点写成类模板…

解决Win10虚拟机“网络连接不上”,“Ethernet0 网络电缆被拔出”的问题

一、情景引入 今天用Win10虚拟机打开浏览器发现&#xff1a; 很奇怪&#xff0c;平常都没有这个问题。 二、检查网络状态 点击更改适配器选项&#xff0c;发现如下&#xff1a; 三、解决问题 打开任务管理器&#xff0c;点击服务&#xff0c;搜索栏搜索&#xff1a;VM …

2025蓝桥杯省赛网络安全组wp

文章目录 黑客密室逃脱ezEvtxflowzipEnigma星际xml解析器EBC-TrainAES-CBC 黑客密室逃脱 提示猜文件名&#xff0c;猜几个常见的&#xff0c;app.py读到源码 这里也是脑抽了一下&#xff0c;把密钥看成1236了。。。卡了五分钟左右&#xff0c;解出来的时候已经降到300多分了&a…

算法查找目录

1. 基础数据结构 数组与链表 动态数组 实现与自动扩容机制均摊分析ArrayList/Vector实现 单向链表 基本操作(插入、删除、查找)链表反转环检测(Floyd判圈算法) 双向链表 插入删除操作优化双向遍历优势边界情况处理 循环链表 约瑟夫环问题单向循环链表双向循环链表 跳表 基本原…

Windows11 管理员用户下无权限操作的解决方法

问题 Program Files 目录下无权限进行写入操作。 Windows 各种功能因权限不足无法访问。 启动某些应用程序时&#xff0c;可能会遇到 用户账户控制 (UAC, User Account Control) 弹窗提示&#xff0c;要求确认是否允许该程序对设备进行更改。 等等问题 解决方法 运行 p…

.NET中,const和readonly区别

在.NET中&#xff0c;const和readonly都用于定义不可变的值&#xff0c;但它们在行为和使用场景上有显著区别。以下是两者的详细对比&#xff1a; 初始化时机 • const ◦ 编译时常量&#xff0c;必须在声明时赋值。 ◦ 值在编译时确定&#xff0c;并被直接嵌入到IL代码中&…

邮件分类特征维度实验分析

活动发起人小虚竹 想对你说&#xff1a; 这是一个以写作博客为目的的创作活动&#xff0c;旨在鼓励大学生博主们挖掘自己的创作潜能&#xff0c;展现自己的写作才华。如果你是一位热爱写作的、想要展现自己创作才华的小伙伴&#xff0c;那么&#xff0c;快来参加吧&#xff01…

数字智慧方案6158丨智慧医疗解决方案精华版(58页PPT)(文末有下载方式)

数字智慧方案6158丨智慧医疗解决方案精华版 详细资料请看本解读文章的最后内容。 引言 随着信息技术的飞速发展&#xff0c;智慧医疗已成为现代医疗体系的重要组成部分。本文将对《数字智慧方案6158丨智慧医疗解决方案精华版》进行详细解读&#xff0c;探讨如何通过先进的技…

Fiori学习专题二十三: Filtering

这节课我们将针对list增加一个筛选功能。 1.首先改造下InvoiceList.view.xml&#xff0c;加入id方便找到它以及标签 <mvc:ViewcontrollerName"ui5.walkthrough.controller.InvoiceList"xmlns"sap.m"xmlns:core"sap.ui.core"xmlns:mvc"s…

大语言模型 05 运行、微调的显存计算详解与优化 全量微调、LoRA 优化策略

写在前面 随着Transformer架构的大语言模型&#xff08;LLM&#xff09;不断发展&#xff0c;其参数规模也在迅速增加。无论是进行模型推理还是微调训练&#xff0c;GPU显存消耗都是开发和应用LLM时的重要考量。本文将详细探讨大模型运行&#xff08;推理&#xff09;与微调时…

对Electron打包的exe文件进行反解析

一、了解 Electron 打包的 exe&#xff0c;本质上就是打包了网页 (HTMLCSSJS)&#xff0c;核心文件是 app.asar。超级容易还原&#xff0c;还原率接近 100% 为什么 Electron 特别容易&#xff1f; 因为 Electron 根本没有真正编译成机器码&#xff0c;它只是把网页资源&…

【Vue2】1-创建一个Vue实例

Vue2官方文档 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title> </head&g…

【C语言练习】015. 声明和初始化指针

015. 声明和初始化指针 015. 声明和初始化指针1. 声明指针示例1:声明一个指向整数的指针2. 初始化指针示例2:将指针初始化为`NULL`示例3:将指针初始化为某个变量的地址示例4:将指针初始化为动态分配的内存地址3. 使用指针访问和修改变量的值示例5:使用指针访问和修改变量的…

好未来golang后端开发

OSI网络模型 TCP和UDP对比 HTTP和HTTPS对比 B树 HTTP常见状态码 线程和进程的区别 goroutine的调度模型GMP 常见的排序了解哪些 快速排序 func quickSort(data []int) {if len(data) < 1 {return}base : data[0]l, r : 0, len(data)-1for i : 1; i < r; {if data[i] &g…

(持续更新)Ubuntu搭建LNMP(Linux + Nginx + MySQL + PHP)环境

LNMP&#xff08;Linux Nginx MySQL PHP&#xff09;环境是在Linux操作系统上构建的一个高性能Web服务器环境。M也可以指代其他数据库&#xff0c;P也可以指代Python 1. 准备Linux系统 确保你已经在一台服务器或虚拟机上安装了Linux操作系统。推荐使用Ubuntu、CentOS或Debi…