当前位置: 首页 > article >正文

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

相关文章:

C语言-指针(一)

目录 指针 内存 概念 指针变量 取地址操作符&#xff08;&&#xff09; 操作符“ * ” 指针变量的大小 注意 指针类型的意义 作用 void * 指针 const修饰指针变量 const放在*前 const放在*后 双重const修饰 指针的运算 1.指针 - 整数 2.指针 - 指针 3.指…...

【DeepMLF】具有可学习标记的多模态语言模型,用于情感分析中的深度融合

这是一篇我完全看不懂的论文,写的好晦涩,适合唬人,所以在方法部分我以大白话为主 abstract 在多模态情感分析(MSA)中,多模态融合已经得到了广泛的研究,但融合深度和多模态容量分配的作用还没有得到充分的研究。在这项工作中,我们将融合深度、可扩展性和专用多模容量作…...

uniapp如何获取安卓原生的Intent对象

通过第三方app唤起&#xff0c;并且获取第三方app唤起时携带的参数 因为应用a唤起应用b时&#xff0c;应用b第一时间就要拿到参数token&#xff0c;所以需要将获取参数的方法写在APP.vue中的onLaunch钩子里,如果其他地方要用可以选择vuex或者采用本地缓存。 uniapp中plus.run…...

implement the “pixel-wise difference“

根据在处理图像数据的来源和格式的不同&#xff0c;在具体实现“两幅图像残差比较”的时候&#xff0c;分为两类方法。 类型一&#xff1a;PyTorch 的 Tensor 图像格式 imgs_pil_o [transforms.ToPILImage()(img_o) for img_o in imgs_o] imgs_pil_w [transforms.ToPILImag…...

tinycudann安装过程加ubuntu18.04gcc版本的升级(成功版!!!!)

使用的是 Linux&#xff0c;安装以下软件包 sudo apt-get install build-essential git安装 CUDA 并将 CUDA 安装添加到您的 PATH。 例如&#xff0c;如果您有 CUDA 12.6.3&#xff0c;请将以下内容添加到您的/usr/local/~/.bashrcexport PATH"/usr/local/cuda-12.6.3/bi…...

Android 实现一个隐私弹窗

效果图如下&#xff1a; 1. 设置同意、退出、点击用户协议、点击隐私协议的函数参数 2. 《用户协议》、《隐私政策》设置成可点击的&#xff0c;且颜色要区分出来 res/layout/dialog_privacy_policy.xml 文件 <?xml version"1.0" encoding"utf-8"?&…...

Oracle无法正常OPEN(三)

在Oracle数据库中&#xff0c;如果几个数据文件丢失&#xff0c;导致数据库无法启动&#xff0c;报错“ORA-01157: cannot identify/lock data file 2 - see DBWR trace file”&#xff0c;如果没有物理备份的情况下&#xff0c;位于丢失数据文件的数据是无法找回的&#xff0c…...

本地服务验证-仙盟创梦IDE-智能编程,编程自动备份+编程审计

本地服务验证server using System; using System.Net;class Program {static void Main(){HttpListener listener new HttpListener();listener.Prefixes.Add("http://localhost:8080/");listener.Start();Console.WriteLine("服务器已启动&#xff0c;监听中…...

[学成在线]22-自动部署项目

自动部署 实战流程 下边使用jenkins实现CI/CD的流程。 1、将代码使用Git托管 2、在jenkins创建任务&#xff0c;从Git拉取代码。 3、拉取代码后进行自动构建&#xff1a;测试、打包、部署。 首先将代码打成镜像包上传到docker私服。 自动创建容器、启动容器。 4、当有代…...

Golang|使用函数作为参数和使用接口的联系

函数作为数据类型的一种&#xff0c;可以成为其他函数的参数。在 Go&#xff08;Golang&#xff09; 中&#xff0c;函数作为参数 和 接口&#xff08;interface&#xff09;&#xff0c;本质上都和抽象、灵活调用有关 —— 都是让代码更灵活、更可扩展的手段。不过它们各有侧重…...

MATLAB技巧——norm和vecnorm两个函数讲解与辨析

在 MATLAB 中,norm 和 vecnorm 是两个用于计算向量或矩阵范数的函数,虽然它们的功能相似,但在使用场景和适用性上存在一些区别。本文将详细解释这两个函数的用途、功能以及如何选择合适的函数。 文章目录 norm函数用法范数类型vecnorm函数用法范数类型选择合适的函数示例对比…...

ubuntu的libc 库被我 sudo apt-get --reinstall install libc6搞没了

我系统的libc 没了 今天为了运行一个开源的yuv 播放器&#xff0c;在运行的时候提醒 Inconsistency detected by ld.so: dl-call-libc-early-init.c: 37: _dl_call_libc_early_init: Assertion sym ! NULL failed!然后听从AI 的建议 当我去执行ls 时&#xff0c;系统提示 就这…...

Ubuntu搭建Conda+Python开发环境

目录 一、环境说明 1、测试环境为ubuntu24.04.1 2、更新系统环境 3、安装wget工具 4、下载miniconda安装脚本 二、安装步骤 1、安装miniconda 2、source conda 3、验证版本 4、配置pip源 三、conda用法 1、常用指令 一、环境说明 1、测试环境为ubuntu24.04.1 2、更…...

智能工厂规划学习——深入解读数字化工厂规划与建设方案

项目总体思路聚焦于通过智能制造和数字化工厂建设,来优化企业战略并提升信息化水平。首先,企业需学习先进国家已经验证的先进经验,并紧跟其正在变革的方向,以确保自身发展的前瞻性和竞争力。 在企业战略层面,企业正从以产品为中心的业务模式,逐步转变为以服务中心…...

【学习笔记】深入理解Java虚拟机学习笔记——第2章 Java内存区域与内存溢出异常

第2章 Java内存区域与内存溢出异常 2.1 概述 略 2.2 运行时数据区域 2.2.1 程序计数器 线程私有&#xff0c;记录执行的字节码位置 2.2.2 Java 虚拟机栈 线程私有&#xff0c;存储一个一个的栈帧&#xff0c;通过栈帧的出入栈来控制方法执行。 -栈帧&#xff1a;对应一个…...

Python全流程开发实战:基于IMAP协议安全下载个人Gmail邮箱内所有PDF附件

在日常办公场景中&#xff0c;面对成百上千封携带PDF附件的邮件&#xff0c;手动逐一下载往往耗时耗力&#xff0c;成为效率瓶颈。如何通过代码实现“一键批量下载”&#xff1f;本文将以**“Gmail全量PDF附件下载工具”**开发为例&#xff0c;完整拆解从需求分析到落地交付的P…...

【验证技能】VIP项目大总结

VIP项目快做一段落了&#xff0c;历时一年半&#xff0c;也该要一个大汇总。 VIP简介 VIP开发流程 VIP难点 进程同步 打拍插入不同bit位宽数据问题。 动态升降lane VIP做的不好的地方和改进想法 各层之间交互 testsuite两端关键 所有层的实现架构不统一 VIP经验 ** 架构…...

Pytest-mark使用详解(跳过、标记、参数 化)

1.前言 在工作中我们经常使用pytest.mark.XXXX进行装饰器修饰&#xff0c;后面的XXX的不同&#xff0c;在pytest中有不同的作 用&#xff0c;其整体使用相对复杂&#xff0c;我们单独将其抽取出来做详细的讲解。 2.pytest.mark.skip()/skipif()跳过用例 import pytest #无条…...

【浅尝Java】Java简介第一个Java程序(含JDK、JRE与JVM关系、javcdoc的使用)

&#x1f35e;自我激励&#xff1a;每天努力一点点&#xff0c;技术变化看得见 文章目录 Java语言概述Java是什么Java语言的重要性Java语言发展简史Java语言特性 第一个Java程序main方法示例运行Java程序JDK、JRE、JVM之间的关系注释基本规则注释规范 标识符关键字 Java语言概述…...

游戏打击感实现

视觉表现 1.帧冻结&#xff08;卡肉&#xff09; 原理&#xff1a;在攻击命中的瞬间暂停动画播放&#xff08;通常0.1-0.3s&#xff09;&#xff0c;伯尼真实打击时的反作用力停滞感。实现&#xff1a;通过控制动画播放速度&#xff08;如Unity的Animator.speed&#xff09;结…...

项目三 - 任务2:创建笔记本电脑类(一爹多叔)

在本次实战中&#xff0c;我们通过Java的单根继承和多接口实现特性&#xff0c;设计了一个笔记本电脑类。首先创建了Computer抽象类&#xff0c;提供计算的抽象方法&#xff0c;模拟电脑的基本功能。接着定义了NetCard和USB两个接口&#xff0c;分别包含连接网络和USB设备的抽象…...

Electron学习+打包

1. 什么是 Electron&#xff1f; Electron 是⼀个 跨平台桌⾯应⽤ 开发框架&#xff0c;开发者可以使⽤&#xff1a;HTML、CSS、JavaScript 等 Web 技术来构建桌⾯应⽤程序&#xff0c;它的本质是结合了 Chromium 和 Node.js &#xff0c;现在⼴泛⽤于桌⾯应 ⽤程序开发&a…...

NumPy线性代数功能全解析:矩阵运算与方程求解实用指南

NumPy 是线性代数领域中高效的工具。它可以帮助完成矩阵运算和方程求解。本文将介绍 NumPy 中用于线性代数的常用函数。 矩阵乘法 矩阵乘法会根据两个矩阵生成一个新矩阵。具体做法是将第一个矩阵的每一行与第二个矩阵的每一列相乘&#xff0c;并将乘积相加&#xff0c;得到新…...

《RabbitMQ 全面解析:从原理到实战的高性能消息队列指南》

一、RabbitMQ 核心原理与架构 1. 核心组件与工作流程 RabbitMQ 基于 AMQP 协议&#xff0c;核心组件包括 生产者&#xff08;Producer&#xff09;、交换机&#xff08;Exchange&#xff09;、队列&#xff08;Queue&#xff09; 和 消费者&#xff08;Consumer&#xff09;。…...

jenkins slave节点打包报错Failed to create a temp file on

jenkins slave节点打包报错 一、报错信息 FATAL: Unable to produce a script file Also: hudson.remoting.Channel$CallSiteStackTrace: Remote call to slave-83at hudson.remoting.Channel.attachCallSiteStackTrace(Channel.java:1784)at hudson.remoting.UserRequest$…...

计算机视觉——通过 OWL-ViT 实现开放词汇对象检测

介绍 传统的对象检测模型大多是封闭词汇类型,只能识别有限的固定类别。增加新的类别需要大量的注释数据。然而,现实世界中的物体类别几乎无穷无尽,这就需要能够检测未知类别的开放式词汇类型。对比学习(Contrastive Learning)使用成对的图像和语言数据,在这一挑战中备受…...

Android Framework学习二:Activity创建及View绘制流程

文章目录 Window绘制流程Window Manager Service&#xff08;WMS&#xff09;SurfaceSurfaceFlinger 安卓View层次结构ActivityPhoneWindowActivity与PhoneWindow两者之间的关系ViewRootImplDecorViewDecorView 的作用DecorView 的结构总结 Activity创建流程View invalidate调用…...

python如何在深度学习框架目标检测算法使用Yolov8训练道路汽车漆面车漆缺陷数据集 建立基于YOLOv8道路汽车漆面缺陷(划痕)检测系统

基于YOLOv8道路汽车漆面缺陷&#xff08;划痕&#xff09;检测系统 文章目录 1. 安装依赖2. 数据集准备与划分3. 数据预处理4. 配置YOLOv85. 训练和评估模型6. 推理与可视化7. 构建GUI应用程序 道路汽车漆面车漆缺陷检测数据集1221张 1类 汽车漆面缺陷检测YOLO数据集 1221张…...

高性能、云原生的对象存储服务MinIO 详细介绍与案例应用

什么是MinIO&#xff1f; MinIO是一个高性能、云原生的对象存储服务&#xff0c;采用Apache License v2.0开源协议发布。它与Amazon S3云存储服务API兼容&#xff0c;适合构建高性能、可扩展的存储基础设施。支持大规模非结构化数据的存储&#xff0c;适合图片、视频、日志、备…...

Tailwind CSS 响应式设计解析(含示例)

本文内容&#xff1a; Tailwindcss V4 中如何使用响应式设计功能&#xff0c;包括默认断点、自定义断点、断点范围控制以及容器查询的各种技巧&#xff0c;帮助你在不离开 HTML 的前提下优雅构建响应式页面。 &#x1f31f; 默认断点用法&#xff08;移动优先&#xff09; Tail…...