C语言实验4:指针

目录

一、实验要求

二、实验原理

1. 指针的基本概念

1.1 指针的定义

1.2 取地址运算符(&)

1.3 间接引用运算符(*)

2. 指针的基本操作

2.1 指针的赋值

2.2 空指针

3. 指针和数组

3.1 数组和指针的关系

 3.2 指针和数组的结合

4. 指针和函数

4.1 指针作为函数参数

5. 动态内存分配

5.1 malloc 和 free 函数

三、实验内容

3.1

代码

截图

分析

3.2

代码

截图

分析


一、实验要求

  1. 掌握指针和间接访问的概念,会定义和使用指针变量。
  2. 能正确使用数组的指针和指向数组的指针变量。
  3. 能正确使用字符串的指针和指向字符串的指针变量。

二、实验原理

指针是C语言中非常重要且强大的概念之一。它提供了直接访问内存地址的能力,使得程序可以更加灵活地处理数据。

1. 指针的基本概念

1.1 指针的定义

 指针是一个变量,其值是另一个变量的地址。通过指针,可以直接访问存储在该地址上的数据。

int *ptr;  // 定义一个指向整数的指针

1.2 取地址运算符(&)

用于获取变量的地址。

int num = 10;
int *ptr = #  // ptr指向num的地址

例如

#include<iostream>
using namespace std;
int main() {int a=10,*ptr;ptr = &a;cout << a <<" "<<ptr;return 0;
}

它的结果是

表面a的地址为000000ECA8AFF794,共16*4=64位二进制

那么下面的代码为什么地址不相邻呢?

#include<iostream>
using namespace std;
int main() {int a=10,b=11,*ptr,*ptr1;ptr = &a;ptr1 = &b;cout << a <<" "<<ptr<<endl;cout << b << " " << ptr1;return 0;
}

在C语言中,连续定义的两个变量的地址是否相邻,与多个因素有关,其中包括编译器、优化选项、操作系统的内存分配策略等。

  1. 对齐(Alignment): 许多体系结构要求数据按照某种规定的边界对齐,以提高访问速度。因此,编译器可能会在变量之间插入填充字节,以确保数据按照正确的边界对齐。这导致即使两个变量类型相同,它们的地址也可能不相邻。

  2. 优化: 编译器可能会对代码进行优化,包括对变量的存储和访问进行优化。这可能导致变量的地址不是按照它们在代码中的声明顺序来分配的。

  3. 内存分配策略: 操作系统对于内存的分配策略也可能影响变量的地址分布。例如,在某些情况下,操作系统可能会使用随机化技术来增加系统的安全性,这会导致变量的地址不再是连续的。

  4. 数据类型: 如果定义的变量类型不同,它们的大小可能也不同,这会影响它们在内存中的布局。

1.3 间接引用运算符(*)

用于访问指针所指向地址上的值。 

int value = *ptr;  // value等于num的值,通过ptr间接引用

例如

#include<iostream>
using namespace std;
int main() {int a=10,*ptr,value;ptr = &a;value = *ptr;cout << a <<" "<<ptr<<"  " <<value<< endl;return 0;
}

结果为

即指针ptr为地址,*ptr代表一个数,&a代表a的地址,&a和ptr可以交换值,属于同一类型

*ptr代表数,a代表数,两者可以交换值,属于同一类型

2. 指针的基本操作

2.1 指针的赋值

可以将一个指针指向另一个变量的地址。

int num1 = 10;
int num2 = 20;
int *ptr = &num1;  // ptr指向num1的地址
ptr = &num2;      // ptr现在指向num2的地址

例如

#include<iostream>
using namespace std;
int main() {int num1 = 10, num2 = 20;int* ptr = &num1;cout << *ptr <<" "<<ptr<< endl;ptr = &num2;cout << *ptr << " " << ptr;return 0;
}

结果为

2.2 空指针

指向地址为0的指针被称为空指针。

int *ptr = NULL;  // ptr是一个空指针

可以猜测一下下述代码的输出结果是什么

#include<iostream>
using namespace std;
int main() {int* ptr = NULL;cout << *ptr <<" "<<ptr<< endl;return 0;
}

答案是!

那空指针有什么作用呢?

1.标记未初始化的指针: 在定义指针变量但尚未为其分配有效内存地址之前,可以将指针初始化为NULL,表示它当前不指向任何有效的内存区域。

2.避免野指针: 将指针初始化为NULL可以避免使用未初始化的指针(野指针),从而减少程序中出现的错误。

3.指针作为空指针常量:NULL表示空指针常量,提高代码的可读性。在函数参数或返回值中,空指针常常用于表示某个指针不指向有效的内存。

void process_data(int *data) {if (data != NULL) {// 处理有效的数据} else {// 处理空指针情况}
}

4.动态内存分配失败的标志: 在动态内存分配时,如果分配失败,malloccalloc通常会返回NULL,这可用于检测内存分配是否成功。

int *dynamicPtr = (int*)malloc(sizeof(int));
if (dynamicPtr == NULL) {// 内存分配失败// 处理错误的代码
}

5.函数返回空指针: 有时,函数可能返回一个空指针作为错误或特殊情况的标志。

int *find_element(int key) {// 查找元素...if (element_not_found) {return NULL;}// 找到元素,返回指向元素的指针
}

3. 指针和数组

3.1 数组和指针的关系

数组名本身就是一个指针,指向数组的第一个元素的地址。

int arr[5] = {1, 2, 3, 4, 5};
int *ptr = arr;  // ptr指向arr的第一个元素

记住是arr为地址,不是&arr! 

 3.2 指针和数组的结合

通过指针可以遍历数组的元素。

for (int i = 0; i < 5; ++i) {printf("%d ", *(ptr + i));  // 打印数组元素的值
}

例如

#include<iostream>
using namespace std;
int main() {int* ptr;int a[5] = { 1,2,3,4,5 };ptr = a;for (int i = 0; i < 5; i++) {cout << "第" << i << "个元素为" << *(a + i) << "其地址为"<<a+i<<endl;}return 0;
}

结果为 

4. 指针和函数

4.1 指针作为函数参数

可以通过指针在函数间传递数据。

void modifyValue(int *ptr) {*ptr = 100;
}int main() {int num = 10;modifyValue(&num);// 现在num的值变成了100return 0;
}

但是如果不用指针呢?

#include<iostream>
using namespace std;
void modifyValue(int ptr) {ptr = 100;
}
int main() {int num = 10;modifyValue(num);cout << num;return 0;
}

结果是10,所以指针的作用就体现出来了

在这段代码中,问题出现在modifyValue函数的参数类型和传递方式上。在C中,函数参数可以通过值传递或引用传递。在这里,modifyValue函数使用的是值传递,这意味着函数接收到的是实参的一个副本而不是实参本身。

当你调用modifyValue(num)时,将num的值(10)传递给modifyValue函数的形参ptr。在函数内部,ptr被修改为100,但这仅仅是对形参的修改,不会影响实参 num 的值。

所以,当你在main函数中输出num的值时,输出的是原始的值,即10,而不是在modifyValue函数内修改后的值100。

4.2 指针作为函数返回值

函数可以返回指针,使得函数能够返回动态分配的内存。

int* createArray(int size) {int *arr = (int*)malloc(size * sizeof(int));// 初始化数组...return arr;
}

例如

#include<iostream>
using namespace std;
int* createArray(int size) {int* arr = (int*)malloc(size * sizeof(int));return arr;
}int main() {int* ptr;ptr = createArray(5);cout << ptr<<" "<<* ptr;return 0;
}

 结果为

为什么*ptr即数组的第一个值那么奇怪呢?

是因为数组只是分配了空间,并没有赋值

5. 动态内存分配

5.1 mallocfree 函数

用于动态分配和释放内存。

int *ptr = (int*)malloc(sizeof(int));  // 分配一个整数大小的内存
// 使用ptr...
free(ptr);  // 释放内存

sizeof 是一个在 C 和 C++ 等编程语言中常用的操作符,用于获取数据类型或变量在内存中所占用的字节数。sizeof 的语法如下

sizeof(type)
sizeof(expression)

其中,type 是数据类型,而 expression 则是一个表达式或变量。sizeof 返回一个 size_t 类型的值,表示参数所占用的字节数。常用于分配内存。

malloc 是 C 语言中的一个函数,用于动态分配内存。它的名字来源于 "memory allocation"(内存分配)。malloc 函数接受一个参数 size,表示要分配的内存字节数。它返回一个 void 指针,指向分配的内存的起始地址。可以将其转换为适当的类型,以便进行正确的使用。

上述转化为int型指针

free 函数用于释放通过动态内存分配函数(如 malloccallocrealloc 等)分配的内存空间。free 函数接受一个指针 ptr,该指针应该是通过动态内存分配函数分配的内存的起始地址。调用 free 函数会将相应的内存空间标记为可用,以便后续的内存分配操作可以使用该空间。

三、实验内容

3.1

将n个数按输入时的顺序逆序排列,用函数实现。

代码

#include<iostream>
using namespace std;
void sort(int* p, int* q, int n) {for (int i = 0; i < n; i++) {*(q + i) = *(p + n - 1 - i);//将q指针所代表的数组的第n-i个元素赋值给p指针所代表的数组的第i+1个元素}
}
int main() {int a[100],b[100],n,*ptr1,*ptr2;ptr1 = a;ptr2 = b;cout << "请输入所需要输入的元素数目:";cin >> n;for (int i = 0; i < n; i++) {cin >> *(ptr1 + i);}sort(ptr1, ptr2, n);for (int i = 0; i < n; i++) {cout << *(ptr2 + i)<<" ";}return 0;
}

截图

分析

  1. void sort(int* p, int* q, int n):这是一个排序函数,接受两个指向整数数组的指针 pq,以及数组的大小 n。该函数通过将数组 p 中的元素逆序复制到数组 q 中来实现排序。

  2. int main():主函数包含以下步骤:

    • 定义两个数组 ab,以及两个指向整数的指针 ptr1ptr2
    • 用户输入所需输入的元素数目 n
    • 通过指针 ptr1 输入数组 a 的元素。
    • 调用 sort 函数,将数组 a 中的元素逆序复制到数组 b 中。
    • 输出数组 b 中的元素,即排序后的结果。
  3. sort 函数中,通过循环遍历数组,将数组 p 中的元素逆序复制到数组 q 中。这通过使用指针算术实现,*(q + i) = *(p + n - 1 - i) 将数组 p 中的第 n-i 个元素赋值给数组 q 中的第 i+1 个元素。

3.2

将一个5×5的矩阵(二维数组)中最大的元素放在中心,4个角分别放4个最小的元素(顺序为从左到右,从上到下依次从小到大存放),写一函数实现。用main函数调用。 

代码

#include<iostream>
using namespace std;
int min_max = -9999, index = 0, b[4] = { 0,4,20,24 };//b数组为快速索引,index为min数组中最大元素的序列号int find_min_max(int* ptr) {//找出四个元素中的最大值min_max = -9999;for (int i = 0; i < 4; i++) {if (*(ptr + i) >= min_max) {min_max = *(ptr + i);index = i;}}return min_max;
}
int compare(const void* a, const void* b) {// 比较函数,用于指定排序规则// 返回负数表示 a < b// 返回零表示 a == b// 返回正数表示 a > breturn (*(int*)a - *(int*)b);
}void change(int *ptr) {//找到最大元素和最小的四个元素int max = -9999;int min[4] = { 9999,9999,9999,9999 };int* ptr2=min;//找最大元素和最小的四个元素for (int i = 0; i < 25; i++) {if (*(ptr + i) > max) {max = *(ptr + i);}min_max = find_min_max(ptr2);if (*(ptr + i) < min_max) {//满足进入四个最小元素的数组*(ptr2 + index) = *(ptr + i);//替换位置}}//对四个最小元素的位置进行排序qsort(ptr2, 4, sizeof(int), compare);//更换位置for (int i = 0; i < 25; i++) {if (*(ptr + i) == max) {//如果是最大元素int temp = *(ptr + 12);*(ptr + 12) = max;*(ptr + i) = temp;continue;}}for (int i = 0; i < 25; i++) {for (int j = 0; j < 4; j++) {//如果是较小元素if (*(ptr + i) == min[j]) {int temp = *(ptr + b[j]);cout << "temp  " << temp << endl;*(ptr + b[j]) = min[j];*(ptr + i) = temp;continue;}			}}	
}
int main() {int a[5][5],* ptr1;ptr1 = a[0];for (int i = 0; i < 5; i++) {for (int j = 0; j < 5; j++) {cin >> a[i][j];}}change(ptr1);for (int i = 0; i < 5; i++) {for (int j = 0; j < 5; j++) {cout<< a[i][j]<<" ";}cout << endl;}return 0;
}

截图

分析

  1. 变量定义和初始化:

    • int min_max = -9999, index = 0, b[4] = { 0,4,20,24 };:定义了全局变量,其中 min_max 用于存储四个元素中的最大值,index 用于存储 min 数组中最大元素的序列号,b 数组是用于快速索引的数组。
  2. find_min_max 函数:

    • 该函数用于找出四个元素中的最大值,并返回这个最大值。它还会更新 index 变量,以表示最大值在 min 数组中的位置。
  3. compare 函数:

    • 这是一个比较函数,用于在后续的 qsort 函数中进行数组排序。
  4. change 函数:

    • 该函数对数组进行操作,找到数组中的最大元素和最小的四个元素,然后将它们的位置进行调整。
    • 具体操作:
      • 找到数组中的最大元素 max
      • 找到数组中最小的四个元素,用 min 数组保存,并通过调用 find_min_max 函数找到最小元素中的最大值及其位置。
      • 使用 qsortmin 数组进行排序。
      • 将最大元素和最小元素的位置进行交换。
      • 输出结果。
  5. main 函数:

    • 定义了一个5x5的数组 a 和一个指向该数组的指针 ptr1
    • 通过用户输入给数组 a 赋值。
    • 调用 change 函数对数组进行操作。
    • 输出最终的数组。

指针是C语言中非常强大和灵活的特性,但也需要小心使用,因为错误的指针操作可能导致程序崩溃或产生不可预测的结果。

指针需慎用! 

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

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

相关文章

【Linux】内核编译 镜像制作

文章目录 一、Ubuntu内核编译1.1 为什么自己编译内核1.2 Ubuntu 内核源码下载1.21 内核的作用1.22 Linux内核与ubuntu内核1.23 Ubuntu内核源码获取 1.3 在Windows系统下编译ubuntu内核1.4 在Linux系统下编译ubuntu内核 二、镜像制作 一、Ubuntu内核编译 1.1 为什么自己编译内核…

用LCD循环右移显示“Welcome to China“

#include<reg51.h> //包含单片机寄存器的头文件 #include<intrins.h> //包含_nop_()函数定义的头文件 sbit RSP2^0; //寄存器选择位&#xff0c;将RS位定义为P2.0引脚 sbit RWP2^1; //读写选择位&#xff0c;将RW位定义为P2.1引脚 sbit EP2^2; //使能…

Debezium日常分享系列之:向 Debezium 连接器发送信号

Debezium日常分享系列之&#xff1a;向 Debezium 连接器发送信号 一、概述二、激活源信号通道三、信令数据集合的结构四、创建信令数据集合五、激活kafka信号通道六、数据格式七、激活JMX信号通道八、自定义信令通道九、Debezium 核心模块依赖项十、部署自定义信令通道十一、信…

【C# 技术】 C# 常用排序方式——自定义数据排序

C# 常用排序方式——自定义数据排序 前言 在最近的项目中经常会对C#中的数据进行排序&#xff0c;对于基本数据类型&#xff0c;其排序方式比较简单&#xff0c;只需要调用内置算法即可实现&#xff0c;但对于自定义数据类型以及自定义排序规则的情况实现起来就比较麻烦&…

区分LR(0),SLR(1),LR(1)和LALR(1)

目录 对于LR(0)文法&#xff1a; 对于SLR(1)文法&#xff1a; 对于LR(0)和SLR(1)文法&#xff1a; 对于LR(1)和SLR(1)文法&#xff1a; 对于LALR(1)文法&#xff1a; 例题1&#xff1a; 例题2&#xff1a; 例题3&#xff1a; 例题4&#xff1a; 这几个文法大致的步骤都…

推荐几个贼有意思的开源项目!

这里有实战项目、入门教程、黑科技、开源书籍、大厂开源项目等&#xff0c;涵盖多种编程语言 Python、Java、Go、C/C、Swift...让你在短时间内感受到开源的魅力&#xff0c;对编程产生兴趣&#xff01; C 项目 1、kilo&#xff1a;不到 1 千行代码实现的迷你文本编辑器。该项…

Portraiture4.1汉化版PS磨皮插件(支持原生m1芯片m2)

Portraiture汉化版PS磨皮插件。本期推荐一款全新ai算法ps2024中文汉化版ps磨皮插件Portraiture 4.1.2美颜滤镜安装包最新版ps调整肤色插件! 全新Portraiture 4.1.2版本PS人像修图美颜磨皮插件&#xff0c;升级AI算法&#xff0c;并支持多人及全身磨皮美化模式&#xff0c;推荐…

【嵌入式开发 Linux 常用命令系列 7.3 -- linux 命令行数值计算】

文章目录 linux 命令行数值计算使用 awk使用 bc 命令使用 Bash 的内置算术扩展使用 expr脚本命令实现 linux 命令行数值计算 在 Linux 命令行中&#xff0c;您可以使用多种方法来执行基本的数学运算。以下是一些示例&#xff1a; 使用 awk awk 是一个强大的文本处理工具&…

【C#】知识点实践序列之Lock的输出多线程信息

大家好&#xff0c;我是全栈小5&#xff0c;欢迎来到《小5讲堂之知识点实践序列》文章。 2023年第2篇文章&#xff0c;此篇文章是C#知识点实践序列之Lock知识点&#xff0c;博主能力有限&#xff0c;理解水平有限&#xff0c;若有不对之处望指正&#xff01; 本篇在Lock锁定代码…

【python_数据分组】

对excel按照标签进行分组&#xff0c;例如按照“开票主体和对方公司”进行分组&#xff0c;并获取对应的明细。 表格如下&#xff1a; def main(excel_data):result {}for d in excel_data:if str(d[0])str(d[1]) in result:result[str(d[0])str(d[1])].append([d[0],d[1],…

web component - 使用HTML Templates和Shadow DOM构建现代UI组件

Web Component是一种用于构建可重用的UI组件的技术。它使用标准化的浏览器API&#xff0c;包括Custom Elements、Shadow DOM和HTML Templates来实现组件化开发方式。这些API都是现代浏览器原生支持的&#xff0c;因此不需要引入第三方库或框架即可使用。 在这篇博客中&#xf…

第6课 用window API捕获麦克风数据并加入队列备用

今天是2024年1月1日&#xff0c;新年的第一缕阳光已经普照大地&#xff0c;祝愿看到这篇文章的所有程序员或程序爱好者都能在新的一年里持之以恒&#xff0c;事业有成。 今天也是我加入CSDN的第4100天&#xff0c;但回过头看一看&#xff0c;这么长的时间也没有在CSDN写下几篇…

SpringCloud-高级篇(九)

&#xff08;1&#xff09;Seata高可用 我们学习了Seata的各种用法了&#xff0c;Seata的服务是单节点部署的&#xff0c;这个服务如果挂了&#xff0c;整个事务都没有办法完了&#xff0c;下面我们学习Seata的高可用的知识。 实现高可用&#xff0c;还是比较简单&#xff0c;…

QT上位机开发(抽奖软件)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 用抽奖软件抽奖&#xff0c;是一种很常见的抽奖方式。特别是写这篇文章的时候&#xff0c;正好处于2023年12月31日&#xff0c;也是一年中最后一天…

【华为机试】2023年真题B卷(python)-喊七的次数重排

一、题目 题目描述&#xff1a; 喊7是一个传统的聚会游戏&#xff0c;N个人围成一圈&#xff0c;按顺时针从1到N编号。 编号为1的人从1开始喊数&#xff0c;下一个人喊的数字为上一个人的数字加1&#xff0c;但是当将要喊出来的数字是7的倍数或者数字本身含有7的话&#xff0c;…

新手快速上手掌握基础排序<二>快速排序快速入门

目录 引言 一&#xff1a;快速排序qsort的简介 1.qsort是一个库函数 2.库函数的查询了解方法 3.qsort的具体使用方法 4.qsort函数使用的一些注意点 5.qsort函数的特点 6.代码实现 (1)整数数组的快速排序 &#xff08;2&#xff09;结构体的快速排序&#xff08;学…

使用.Net nanoFramework 驱动ESP32的OLED显示屏

本文介绍如何使用.Net nanoFramework 驱动ESP32的OLED显示屏。我们将会从最基础的部分开始&#xff0c;逐步深入&#xff0c;让你能够理解并实现整个过程。无论你是初学者还是有一定经验的开发者&#xff0c;这篇文章都会对你有所帮助。 1. 硬件准备 1.1 ESP32开发板 这里我们…

PyTorch的Tensor(张量)

一、Tensor概念 什么是张量&#xff1f; 张量是一个多维数组&#xff0c;它是标量、向量、矩阵的高维拓展 Tensor与Variable Variable是torch.autograd中的数据类型&#xff0c;主要用于封装Tensor&#xff0c;进行自动求导。 data: 被包装的Tensorgrad: data的梯度&…

基于OpenCv的车道检测

项目背景和步骤 车道检测是自动驾驶领域不可或缺的一环 具体步骤如下&#xff1a; 一、将图像灰度化&#xff0c;并进行适度的高斯滤波&#xff0c;剔除干扰 二、利用Canny边缘检测&#xff0c;检测出车道和其它物体的边缘 三、使用ROI区域截取&#xff0c;截取需要的部分&a…

AI与数字化映像:颜值开端,功能至上_光点科技

在人工智能的浪潮中&#xff0c;AI数字人的兴起正成为一个不可忽视的现象。随着ChatGPT等生成式AI算法的进步&#xff0c;AIGC&#xff08;人工智能生成内容&#xff09;的应用呈现出爆发性增长&#xff0c;不仅在技术圈引起广泛关注&#xff0c;也为元宇宙及其相关产业链带来了…