0602-指针和数组

news2024/12/24 20:42:42

数组名

数组名是一个地址常量,不允许赋值。它表示数组首元素的地址。

指针操作数组元素

指针访问数组

指针类型变量\常量+1等同于指针保存的内存地址+sizeof(指针指向的数据类型)

2个相同类型的指针相减,得到的结果是2个指针的偏移量。其中偏移单位(也叫步长)为sizeof(指针指向的数据类型)

#include<stdio.h>

int main() {
	int arr[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
	int* p = arr;
	int length = sizeof(arr) / sizeof(int);

	for (size_t i = 0; i < length; i++)
	{
		printf("arr[%d]=%d, p[%d]=%d, *(arr+%d)=%d, *(p+%d)=%d\n", 
			i, arr[i], i, p[i], i, *(arr+i), i, *(p+i));
	}

	int* q = arr;
	for (size_t i = 0; i < length; i++)
	{
		printf("第%d项=%d\n", i, *q);
		q++;
	}

	/*
	下面计算得出,q与arr的偏移量为10(个步长)
	*/
	int step = q - arr;
	printf("step=%d\n", step);
	return 0;
}

运行上面的代码,结果如下:
在这里插入图片描述
PS:指针操作数组时推荐使用p[i],因为p[i]*(p+i)简单明了。
[i]相当于偏移+取值,p[i]就表示以p保存的地址为起点,偏移i个步长并且取值。

数组名与指针的区别

1、数组名是常量,指针是变量;
2、sizeof的结果不同;

#include<stdio.h>

int main() {
	int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
	int* p = arr;

	printf("sizeof(arr) = %u\n", sizeof(arr));
	printf("sizeof(p) = %u\n", sizeof(p));
	return 0;
}

执行上面代码,结果如下:
在这里插入图片描述

数组名退化

数组作为函数参数时,数组名退化为指针变量,丢失数组长度。

#include<stdio.h>

/*
printArrayLen(int arr[])等价于printArrayLen(int* arr)
*/
void printArrayLen(int arr[]) {
	int len = sizeof(arr) / sizeof(arr[0]);
	printf("参数数组的长度:%d\n", len);
	/*
	* 数组作为函数参数时,数组名退化为指针,
	* 所以此处arr相当于指针变量。
	*/
	printf("sizeof(arr) = %d\n", sizeof(arr));

}

int main() {
	int a[10] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
	printArrayLen(a);
	return 0;
}

运行上面代码,结果如下:
在这里插入图片描述

指针加减运算

指针自增或自减的步长只和指针指向的数据类型有关,与实际赋值给指针变量的数据类型无关。

#include<stdio.h>

int main() {
	int arr[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
	//给p和p2赋上相同的值
	int* p = &arr[9];
	char* p2 = &arr[9];
	printf("p保存的地址:%p\n", p);
	printf("p2保存的地址:%p\n", p2);

	p--;
	p--;
	p--;
	p2--;
	p2--;
	p2--;
	//p自减的步长为4字节,p2自减的步长为1字节
	printf("\"p--\"3次后, p保存的地址:%p\n", p);
	printf("\"p2--\"3次后, p2保存的地址:%p\n", p2);
}

运行上面代码,结果如下:
在这里插入图片描述
指针的运算符,仅支持“指针与整数的加减运算”、“同类型指针相减”、“两指针比大小”、“逻辑运算符”,其余指针运算都是没有意义的。

指针数组

指针数组是一个数组,它的每一个元素都是指针类型。

#include<stdio.h>

void main() {
	int a = 10;
	int b = 20;
	int c = 30;

	int* arr[3] = {&a, &b, &c};

	printf("a = %d\n", *arr[0]);
	return 0;
}

运行上面代码,结果如下:
在这里插入图片描述
指针数组相当于一个二维数组。

#include<stdio.h>

void main() {

	int a[] = { 1, 2, 3 };
	int b[] = { 4, 5, 6 };
	int c[] = { 7, 8, 9 };

	int* d[] = {a, b, c};

	for (int i = 0; i < sizeof(d) / sizeof(d[0]); i++) {
		printf("%d %d %d\n", *d[i], *(d[i]+1), *(d[i]+2));
	}
	printf("=====\n");
	for (int i = 0; i < sizeof(d) / sizeof(d[0]); i++) {
		printf("%d %d %d\n", d[i][0], d[i][1], d[i][2]);
	}
	printf("=====\n");
	for (int i = 0; i < sizeof(d) / sizeof(d[0]); i++) {
		printf("%d %d %d\n", (*(d+i))[0], (*(d + i))[1], (*(d + i))[2]);
	}
}

运行上面代码,结果如下:
在这里插入图片描述

多级指针

#include<stdio.h>

int main() {

	int a[] = {1, 2, 3};
	int b[] = {4, 5, 6};
	int c[] = {7, 8, 9};

	int* arr[] = {a, b, c};
	/*
	p保存的是&(int*),也就是int*型变量的地址,而不是int*型变量的值
	arr是数组首元素a的地址,而a类型是int*,所以arr是int*的地址。
	所以可以将arr赋值给p。

	从而指针数组和二级指针建立关系。
	二级指针指向指针数组的首元素。
	*/
	int** p = arr;

	printf("p最终指向的整形为:%d\n", **p);
	printf("*p = %p\n", *p);
	printf("数组a的首地址=%p\n", a);
	printf(" *(p+1) = %p\n", *(p+1));
	printf("数组b的首地址=%p\n", b);
	printf(" *(p+2) = %p\n", *(p + 2));
	printf("数组c的首地址=%p\n", c);

	printf("====遍历指针数组====\n");
	for (int i = 0; i < 3; i++) {
		for (int j = 0; j < 3; j++) {
			//下面三行代码等价
			printf("%d ", p[i][j]);
			//printf("%d ", *(p[i]+j));
			//printf("%d ", *(*(p + i) + j));
		}
		printf("\n");
	}
	return 0;
}

运行上面代码,结果如下
在这里插入图片描述

#include<stdio.h>

int main() {
	int a = 10;
	int b = 20;
	int* p = &a;
	int** pp = &p;

	//下面2行代码输出“*p = 20”
	//*pp = &b;
	//printf("*p = %d\n", *p);


	//下面3行代码输出
	//*p = 30
	//a = 30
	**pp = 30;
	printf("*p = %d\n", *p);
	printf("a = %d\n", a);
	return 0;
}
int main() {
	int a = 10;
	int* p = &a;
	int** pp = &p;
	int*** ppp = &pp;

	//下列等式成立
	//*ppp == pp == &p
	//**ppp == *pp == p == &a
	//***ppp == **pp == *p == a
}

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

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

相关文章

STL模型转有限元网格

随着计算机图形学、硬件和3D打印技术的发展&#xff0c;基于曲面三角网格的图形渲染及其文件格式越来越流行。 在有限元分析&#xff08;FEA&#xff09;中&#xff0c;工程师有时会得到基于三角形网格的几何文件&#xff08;如STL文件等&#xff09;并进行后续分析。 由于曲面…

华为OD机试真题 Java 实现【检查是否存在满足条件的数字组合】【2022Q4 100分】

一、题目描述 给定一个正整数数组&#xff0c;检查数组中是否存在满足规则的数字组合 规则&#xff1a;A B 2C 二、输入描述 第一行输出数组的元素个数。 接下来一行输出所有数组元素&#xff0c;用空格隔开。 三、输出描述 如果存在满足要求的数&#xff0c;在同一行…

【MySQL】不允许你不会SQL语句之DDL

目录 前言&#xff1a; 一.DDL数据库语句 1.1语句讲解 1.2总结 二.DDL表语句 2.1语句讲解 2.2总结 三.DDL字段语句 3.1语句讲解 3.2总结 四.MySQL数据类型 五.结尾 前言&#xff1a; 在从零到一入门MySQL一篇中&#xff0c;我们对数据库已经有了一定的了解&#xf…

Linux 设备树手动反编译 dtb 生成 源文件dts

验证平台 win10 64 位 VMware Workstation Pro 16 ubuntu 20.04 dtc 工具&#xff1a; 来自 linux-6.3.5 中的 scripts/dtc/ 安装 dtc 工具 其实可以通过安装 dtc 软件包&#xff0c;但不建议这么做&#xff0c;最好通过编译Linux 最新的内核&#xff0c;获取这个 dtc 工具…

利用Web Serial API实现Vue与单片机串口通信

一、Web Serial API介绍 Web Serial API 是一项 Web 技术&#xff0c;用于在浏览器中访问串行端口设备&#xff08;如 Arduino、传感器等&#xff09;并与之通信。它提供了一组 JavaScript 接口&#xff0c;使得 Web 应用程序可以通过 USB 串行端口连接到硬件设备&#xff0c;并…

华为OD机试真题 Java 实现【报数游戏】【2022Q4 100分】

一、题目描述 100个人围成一圈&#xff0c;每个人有一个编码&#xff0c;编号从1开始到100。他们从1开始依次报数&#xff0c;报到为M的人自动退出圈圈&#xff0c;然后下一个人接着从1开始报数&#xff0c;直到剩余的人数小于M。请问最后剩余的人在原先的编号为多少&#xff…

【JavaSE】Java基础语法(四十五):TCP UDP 全解

文章目录 1. TCP发送数据2. TCP接收数据【应用】3. TCP程序练习4. TCP程序文件上传练习【应用】5. UDP发送数据6. UDP接收数据【应用】7. UDP通信程序练习【应用】8. UDP三种通讯方式 1. TCP发送数据 Java中的TCP通信 Java对基于TCP协议的的网络提供了良好的封装&#xff0c;使…

chatgpt赋能python:Python删除文件目录

Python删除文件目录 Python是一种高级编程语言&#xff0c;广泛应用于开发各种类型的应用程序。Python的许多功能使其成为开发者的首选编程语言之一。在这篇文章中&#xff0c;我们将讨论如何使用Python删除文件和目录。 文件和目录的区别 在开始之前&#xff0c;让我们了解…

16_Linux内核启动流程

目录 链接脚本vmlinux.Ids Linux内核入口stext mmap_switched函数 start_kernel函数 rest_init函数 init进程 链接脚本vmlinux.Ids 要分析Linux启动流程,同样需要先编译一下Linux源码,因为有很多文件是需要编译才会生成的。首先分析Linux内核的连接脚本文件arch/arm/kem…

【Python】Python系列教程-- Python3 字符串(十一)

文章目录 前言Python3 字符串Python 访问字符串中的值Python 字符串更新Python 转义字符Python 字符串运算符Python 字符串格式化Python三引号f-stringUnicode 字符串Python 的字符串内建函数 前言 往期回顾&#xff1a; Python系列教程–Python3介绍&#xff08;一&#xff…

计算机网络-网络层1.0

传输单位 网络层将分组从源端传到目的端&#xff0c;为分组交换网上的不同主机提供通信服务&#xff0c;传输单位为数据报 路由器 完成路由选择和分组转发 路由选择&#xff1a;按照分布式算法&#xff0c;根据从相邻路由器所得到的的关于整个网络拓扑的变化情况&#xff0…

操作系统第二章​练习题

第一部分 教材习题&#xff08;P84&#xff09; 1、什么是前趋图&#xff1f;为什么要引入前趋图&#xff1f;​ 前趋图:是用于描述程序段或进程之间执行的先后顺序的有向无循环图。 引入的原因:为了更好的描述程序的顺序和并发执行的情况。 试画出下面4条语句的前趋图&#…

LeetCode 2559 统计范围内的元音字符串数

LeetCode 2559 统计范围内的元音字符串数 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/count-vowel-strings-in-ranges/description/ 博主Github&#xff1a;https://github.com/GDUT-Rp/LeetCode 题目&#xff1a;…

Vue3小兔鲜:组合式写法入门

Vue3&#xff1a;组合式写法入门 Date: May 11, 2023 认识Vue3 1. Vue3组合式API体验 通过 Counter 案例 体验Vue3新引入的组合式API <script> export default {data(){return {count:0}},methods:{addCount(){this.count}} } </script><script setup> imp…

【JavaSE】Java基础语法(四十四):XML解析

文章目录 1. 概述2.标签的规则3. 语法规则【应用】4. xml解析【应用】 1. 概述 万维网联盟(W3C) 万维网联盟(W3C)创建于1994年&#xff0c;又称W3C理事会。1994年10月在麻省理工学院计算机科学实验室成立。 建立者&#xff1a; Tim Berners-Lee (蒂姆伯纳斯李)。 是Web技术领域…

【分布族谱】高斯分布和逆高斯分布的关系

文章目录 高斯分布逆高斯分布简介通过高斯分布构造逆高斯分布 高斯分布 正态分布&#xff0c;又称Gauss分布&#xff0c;其概率密度函数入下图所示 正态分布 N ( μ , σ ) N(\mu, \sigma) N(μ,σ)受到期望 μ \mu μ和方差 σ 2 \sigma^2 σ2的调控&#xff0c;其概率密度函…

基于SpringBoot+Vue的医疗服务系统设计与实现

博主介绍&#xff1a; 大家好&#xff0c;我是一名在Java圈混迹十余年的程序员&#xff0c;精通Java编程语言&#xff0c;同时也熟练掌握微信小程序、Python和Android等技术&#xff0c;能够为大家提供全方位的技术支持和交流。 我擅长在JavaWeb、SSH、SSM、SpringBoot等框架下…

Qt for Android环境配置(联合Android Studio)

目录 1.安装JDK2.安装Android Studio并下载必要组件3.安装QtCreator并配置 Official Doc 1.安装JDK Oracle JDK Download 配置环境变量&#xff1a; 2.安装Android Studio并下载必要组件 Android Studio Download 下图中&#xff0c;31.0.0是当前Qt必需的&#xff0c;3…

chatgpt赋能python:Python制作抽奖——让你的活动更有趣

Python制作抽奖——让你的活动更有趣 在活动中&#xff0c;抽奖环节常常是一大亮点。而用Python制作抽奖程序&#xff0c;则可以更灵活、更实用地完成这个环节。接下来我们将介绍Python制作抽奖的方法&#xff0c;并且说明如何让你的抽奖环节更加有趣。 Python制作抽奖的方法…

【Linux网络服务】Nginx优化

Nginx网页优化 一、配置Nginx网页缓存时间1.1设置方法 二、隐藏Nginx版本号2.1方法一&#xff1a;修改配置文件2.2方法二&#xff1a;修改源码文件&#xff0c;重新编译 三、修改用户与组四、日志切割五、连接超时六、更改进程数七、网页压缩八、配置防盗链九、Linux内核参数优…