【C语言】带你完全理解指针(五)练习

news2024/11/26 12:16:24

复习一下对数组名的理解

数组名的理解
数组名是数组首元素的地址
但是有2个例外:
1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节
2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址

一维数组

int a[] = { 1,2,3,4 };
printf("%d\n", sizeof(a));
printf("%d\n", sizeof(a + 0));
printf("%d\n", sizeof(*a));
printf("%d\n", sizeof(a + 1));
printf("%d\n", sizeof(a[1]));
printf("%d\n", sizeof(&a));
printf("%d\n", sizeof(*&a));
printf("%d\n", sizeof(&a + 1));
printf("%d\n", sizeof(&a[0]));
printf("%d\n", sizeof(&a[0] + 1));

在32位下的结果是4 4 4/8 

  讲解:

printf("%d\n", sizeof(a));

sizeof(数组名),数组名单独放在sizeof中,所以计算的是整个数组的大小,4*4=16。

printf("%d\n", sizeof(a + 0));

 a是数组名是首元素地址,a+0(不是单独放在sizeof中)还是首元素地址,地址的大小是4/8。

printf("%d\n", sizeof(*a));

 数组名a是数组首元素的地址,*a就是首元素,大小就是4个字节

printf("%d\n",sizeof(a + 1));

数组名a是数组首元素的地址,a+1是第二个元素的地址,地址的大小4/8

printf("%d\n".sizeof(a[1]));

第二个元素的大小就是4个字节(int类型大小)

printf("%d\n", sizeof(&a));

 &a是数组的地址,数组的地址也是地址,是地址4/8个字节

printf("%d\n", sizeof(*&a));

计算整个数组的大小所以是16个字节。

sizeof(*&a)-->sizeof(a)两者等价

printf("%d\n", sizeof(&a + 1));

&a+1相当于&a是跳过整个数组,但是即使是跳过整个数组,&a+1依然是地址,是地址就是4/8个字节

printf("%d\n",sizeof(&a[0]));

在C语言中[](数组下标运算符)的优先级高于&(取地址运算符)。所以是先a[0]就是第一个元素,之后&,得到第一个元素的地址就是4/8.也就是第一个元素的地址。

printf("%d\n", sizeof(&a[0] + 1));

&a[0]是首元素的地址,&a[0]+1就是第二个元素的地址,大小4。

 字符数组

char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));
printf("%d\n", sizeof(arr+0));
printf("%d\n", sizeof(*arr));
printf("%d\n", sizeof(arr[1]));
printf("%d\n", sizeof(&arr));
printf("%d\n", sizeof(&arr+1));
printf("%d\n", sizeof(&arr[0]+1));

 (32位下)【答案】:6 4 1 1 4 4 4

【解析】

printf("%d\n", sizeof(arr));

  数组名arr单独放在sizeof内部,计算的是整个数组的大小6。

printf("%d\n", sizeof(arr + 0));

 arr表示数组首元素的地址,arr+0还是数组首元素的地址,是地址就是4/8个字节

printf("%d\n", sizeof(*arr));

arr表示数组首元素的地址,*arr就是首元素,大小1个字节

  

printf("%d\n", sizeof(arr[1]));

arr[1]就是第二个元素,大小是1个字节

printf("%d\n", sizeof(&arr));

&arr是数组的地址,但是数组的地址也是地址,是地址就是4/8

printf("%d\n", sizeof(&arr + 1));

 &arr+1 是跳过数组后的地址, 是地址就是4。

printf("%d\n", sizeof(&arr[0] + 1));

  第二个元素的地址,是地址就是4。

看看strlen的结果,注意strlen统计的是\0之前的字符个数,现在这个数组中是没有\0的

printf("%d\n", strlen(arr));
printf("%d\n", strlen(arr+0));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
printf("%d\n", strlen(&arr));
printf("%d\n", strlen(&arr+1));
printf("%d\n", strlen(&arr[0]+1));

 【答案】32位环境下运行的结果

1、随机值

2、随机值

3、err

4、err

5、随机值

6、随机值-6

7、随机值-1

 【解析】

printf("%d\n", strlen(arr));

 因为字符数组arr中没有\0,所以在求字符串长度的时候,会一直往后找,产生的结构就是随机值

printf("%d\n", strlen(arr + 0));

arr + 0是首元素的地址,和第一个一样,也是随机值。strlen函数是接收一个地址然后往后面找\0。也是随机值

printf("%d\n", strlen(*arr));

 arr是数组首元素的地址,*arr就是数组首元素,就是'a'-97

strlen函数参数的部分需要传一个地址,当我们传递的是'a'时,'a'的ASCII码值是97,那就是将97作为地址传参

strlen就会从97这个地址开始统计字符串长度,这就非法访问内存了

printf("%d\n", strlen(arr[1]));

 同上

printf("%d\n", strlen(&arr + 1));

 &arr表示整个数组的地址,&arr + 1表示跳过整个数组(6个字节)向后查找\0,因此是随机值

printf("%d\n", strlen(&arr[0] + 1));

 第二个函数的地址。也是随机值。 &arr[0]表示首元素地址,&arr[0] + 1跳过一个元素,向后查找\0,因此是随机值

下面换成char arr[] = "abcdef";再来看看也就是末尾有\0了

char arr[] = "abcdef";
printf("%d\n", sizeof(arr));
printf("%d\n", sizeof(arr+0));
printf("%d\n", sizeof(*arr));
printf("%d\n", sizeof(arr[1]));
printf("%d\n", sizeof(&arr));
printf("%d\n", sizeof(&arr+1));
printf("%d\n", sizeof(&arr[0]+1));


printf("%d\n", strlen(arr));
printf("%d\n", strlen(arr+0));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
printf("%d\n", strlen(&arr));
printf("%d\n", strlen(&arr+1));
printf("%d\n", strlen(&arr[0]+1));


 

 【答案】32位环境下运行的结果

7

4

1

1

4

4

4



1、6

2、6

3、err

4、err

5、6

6、随机值

7、5

 【解析】

printf("%d\n", sizeof(arr));

 sizeof(数组名)表示整个数组,sizeof是会把\0也计入在内的,因此是7。 


printf("%d\n", sizeof(arr + 0));

 arr+0表示首元素地址,是地址就是4/8


printf("%d\n", sizeof(*arr));

 *arr表示首元素,首元素是char类型,所以就是1。 


printf("%d\n", sizeof(arr[1]));

 第二个元素,所以就是1。


printf("%d\n", sizeof(&arr));

&arr表示整个数组的地址, 是地址就是4/8 


printf("%d\n", sizeof(&arr + 1));

 &arr表示整个数组的地址, &arr + 1表示跳过整个数组,是地址就是4。


printf("%d\n", strlen(arr));

 统计\0前有多少个元素,就是6。


printf("%d\n", strlen(arr + 0));

 arr+0等于首元素地址,统计\0前有多少个元素,就是6。


printf("%d\n", strlen(*arr));

 *arr表示首元素,把元素作为地址直接进行访问,就是非法访问,因此程序会报错。 


printf("%d\n", strlen(arr[1]));

同上


printf("%d\n", strlen(&arr));

&arr是数组的地址这里其实数值和首元素地址一样,所以是6


printf("%d\n", strlen(&arr + 1));

跳过这个数组向后找|0,随机值。


printf("%d\n", strlen(&arr[0] + 1));

 &arr[0] + 1 表示第二个元素的地址,向后找\0等于5。



char* p = "abcdef";
//         012345

	printf("%d\n", strlen(p));
	printf("%d\n", strlen(p + 1));
	printf("%d\n", strlen(*p));
	printf("%d\n", strlen(p[0]));
	printf("%d\n", strlen(&p));
	printf("%d\n", strlen(&p + 1));
	printf("%d\n", strlen(&p[0] + 1));

【答案】

 6

5

err

err

随机值

随机值

5

 【解析】

printf("%d\n", strlen(p));//6

前面说了strlen接收的是指针,p就是这个字符串的指针也就是首元素的地址。所以是6

printf("%d\n", strlen(p + 1));

p+1是第二个元素的地址,向后找\0所以是5

printf("%d\n", strlen(*p))

 *p是这个字符串,非法访问。

printf("%d\n", strlen(p[0]))

 同理,这也是非法访问,p[0]是第一个元素。strlen要的是一个指针(地址)

printf("%d\n", strlen(&p));//随机值

这是这个指针的地址。

printf("%d\n", strlen(&p + 1));//随机值

printf("%d\n", strlen(&p[0] + 1));//5

 第二个元素开始向后找\0所以是5

char* p = "abcdef";

printf("%d\n", sizeof(p));


      printf("%d\n", sizeof(p + 1));

	printf("%d\n", sizeof(*p));
	printf("%d\n", sizeof(p[0]));

	printf("%d\n", sizeof(&p));
	//&p -- char**
	printf("%d\n", sizeof(&p + 1));

	printf("%d\n", sizeof(&p[0] + 1));

【答案】 

4

4

1

1

4

4

4

 解析

printf("%d\n", sizeof(p));

p是一个指针变量大小就是4/8


printf("%d\n", sizeof(p + 1));

p+1是'b'的地址,是地址大小就是4/8个字节


printf("%d\n", sizeof(*p));

*p 就是'a',就是1个字节


printf("%d\n", sizeof(p[0]));

p[0]--> *(p+0) --> *p 1个字节


printf("%d\n", sizeof(&p));//4/8

&p 的类型是 char**是二级指针,是指针就是4/8


printf("%d\n", sizeof(&p + 1));

&p + 1表示指向&p之后位置的指针。如果指针p在内存中占用4个字节,那么&p + 1将指向p之后的4个字节。

是地址就是4/8


printf("%d\n", sizeof(&p[0] + 1));//4/8 

&p[0] + 1得到是'b'的地址.是地址就是4/8

二维数组

int a[3][4] = {0};
 printf("%d\n",sizeof(a));
 printf("%d\n",sizeof(a[0][0]));
 printf("%d\n",sizeof(a[0]));
 printf("%d\n",sizeof(a[0]+1));
 printf("%d\n",sizeof(*(a[0]+1)));
 printf("%d\n",sizeof(a+1));
 printf("%d\n",sizeof(*(a+1)));
 printf("%d\n",sizeof(&a[0]+1));
 printf("%d\n",sizeof(*(&a[0]+1)));
 printf("%d\n",sizeof(*a));
 printf("%d\n",sizeof(a[3]));

【答案】 

48

4

16

4

4

4

16

4

16

16

16

【解析】

printf("%d\n", sizeof(a));//3*4*4 = 48

计算整个数组的大小是3*4个元素,每个元素是4字节所以是48字节


printf("%d\n", sizeof(a[0][0]));//4

第一个元素


printf("%d\n", sizeof(a[0]));//a[0]是第一行这个一维数组的数组名

a[0]是第一行这个一位数组的数组名。数组名算是单独放在sizeof内部了,计算的是整个数组的大小,大小是16个字节


printf("%d\n", sizeof(a[0] + 1));

a[0]作为第一行的数组名,没有单独放在sizeo内部,没有&

a[0]表示数组首元素的地址,也就是a[0][0]的地址

所以a[0]+1是第一行第二个元素的地址,是地址就是4/8个字节


printf("%d\n", sizeof(*(a[0] + 1)));//4

计算的是就是第一行第2个元素的大小。


注意这个
printf("%d\n", sizeof(a + 1));//4 / 8

a是数组首元素的地址,是第一行的地址 类型就是int(*)[4]。a+1 就是第二行的地址

加减整数的操作移动多少是看自己的类型的,第一行的地址的类型是int(*)[4]是数组指针


printf("%d\n", sizeof(*(a + 1)));//16

a+1就是第二行的地址,*(a+1) --> a[1] -> sizeof(*(a+1))->sizeof(a[1]) 计算的是第二行的大小

a+1 --> 是第二行的地址,int(*)[4]

*(a+1) 访问的第二行的数组


printf("%d\n", sizeof(&a[0] + 1));//4/8

&a[0]是第一行的地址,加一之后就是第二行的地址

&a[0]是第一行的地址 int(*)[4]

&a[0]+1 是第二行的地址 int(*)[4]


printf("%d\n", sizeof(*(&a[0] + 1)));//16 计算的是第二行的大小

看了上一个就知道这里相当数是把第二行的数组名直接放在sizeof

里面所以是第二行的大小16


printf("%d\n", sizeof(*a));//计算的是第一行的大小-16

a,int[*][4],解引用之后就是第一行这个一维数组

a是数组首元素的地址,就是第一行的地址

*a 就是第一行

*a --> *(a+0) --> a[0]


printf("%d\n", sizeof(a[3]));//16

a[3]--> int [4]

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

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

相关文章

《六》输入组控件InputWidget---Combo Box

一、Qt 输入组控件(Input Widgets) Qt Input Widgets是一组用户界面元素,用于输入和显示文字和数字等的数据。这些小部件可以组成各种不同的表单和对话框,用户可以使用这些小部件与程序交互。 以下是Qt Input Widgets的一些常见小部件&…

L2-3 完全二叉树的层序遍历

完全二叉树的层序遍历 一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是完美二叉树。对于深度为 D 的,有 N 个结点的二叉树,若其结点对应于相同深度完美二叉树的层序遍历的前 N 个结点,这样的树就是完全…

跟着Datawhale重学数据结构与算法

数据结构和算法之前学过,现在跟着Datawhale重学一下,就当是监督自己学习,重新拾起来养成一个好的习惯,以后可以一直坚持下去。 开源链接:【 教程地址 】【电子网站】 首先: #mermaid-svg-Cdr3rn9fGCVAiKS…

STM32-ADC(独立模式、双重模式)

ADC简介 18个通道:外部信号源就是16个GPIO回。在引脚上直接接模拟信号就行了,不需要侄何额外的电路。引脚就直接能测电压。2个内部信号源是内部温度传感器和内部参考电压。 逐次逼近型ADC: 它是一个独立的8位逐次逼近型ADC芯片,这个ADC0809是…

C语言 | 自定义类型:联合和枚举

目录: ----前言 1. 联合体 1.1 联合体类型的声明 1.2 联合体的特点 1.3 相同成员的结构体和联合体对比 1.4 联合体大小的计算 1.5 联合的使用 1.6联合体的练习 2. 枚举 2.1 枚举类型的声明 2.2 枚举类型的优点 2.3 枚举类型的使用 --前言: c语言中内…

String类的几个常用方法

描述 以下程序演示了String类的几个常用方法,包括比较字符串、取得字符串长度、拆分字符串、获取子串、字符串转换成字符数组、获取字符串位置、替换字符串等方法。 案例 public class StringTest {public static void main(String[] args) {//比较字符串String …

阿赵UE学习笔记——28、粒子系统Niagara简介

阿赵UE学习笔记目录 大家好,我是阿赵。   继续学习虚幻引擎的使用。这次开始学习粒子系统的使用。 一、Cascade系统 在介绍UE5的Niagara系统之前,必须先介绍一下旧版本的粒子系统。   在UE4的时候,虚幻引擎的粒子系统叫做Cascade&#x…

ubuntu 使用conda 创建虚拟环境总是报HTTP错误,转换多个镜像源之后仍报错

最近在使用Ubuntu conda创建虚拟环境时,总是报Http错误,如下图所示: 开始,我以为是conda 镜像源的问题,但是尝试了好几个镜像源都不行,还是报各种各样的HTTP错误。后来查阅很多,总算解决了。解…

数学建模--深入剖析线性规划(模型全方位解读+代码分析)

1.简介 (1)线性规划三要素 (2)模型适用赛题 2.典例讲解 (1)问题分析 目标函数是净收益尽可能大,风险尽可能小; 约束条件是交易费的分段函数,以及每一笔投资都是非负数&am…

【精读文献】Scientific data|2017-2021年中国10米玉米农田变化制图

论文名称:Mapping annual 10-m maize cropland changes in China during 2017–2021 第一作者及通讯作者:Xingang Li, Ying Qu 第一作者单位及通讯作者单位:北京师范大学地理学部 文章发表期刊:《Scientific data》&#xff08…

hot100 -- 链表(中)

不要觉得力扣核心代码模式麻烦,它确实比不上ACM模式舒服,可以自己处理输入输出 只是你对 链表 和 return 的理解不到位 👂 ▶ 屿前世 (163.com) 👂 ▶ see you tomorrow (163.com) 目录 🎂两数相加 🚩删…

android不同版本(支持>10)获取当前连接的wifi名称

1、AndroidManifest.xml 配置权限 <uses-permission android:name"android.permission.ACCESS_COARSE_LOCATION" /> <uses-permission android:name"android.permission.CHANGE_NETWORK_STATE" /> <uses-permission android:name&q…

Rust腐蚀服务器清档多教程

Rust腐蚀服务器清档多教程 大家好我是艾西&#xff0c;一个做服务器租用的网络架构师。上期教了大家怎么搭建服务器以及安装插件等那么随着大家自己架设服或是玩耍的时间肯定会有小伙伴想要去新增开区数量或是把原本的服务器进行一些调整等&#xff0c;那么今天主要聊的就是怎…

Cohere Command R+:企业级可扩展大模型

Command R 是一种最先进的 RAG 优化模型&#xff0c;旨在处理企业级工作负载&#xff0c;并首先在 Microsoft Azure 上提供 今天&#xff0c;我们推出了 Command R&#xff0c;这是我们最强大、可扩展的大型语言模型 &#xff08;LLM&#xff09;&#xff0c;专为在实际企业用…

tailwindcss在manoca在线编辑智能感知

推荐一下monaco-tailwindcss库&#xff0c;它实现在monaco-editor网页在线编辑器中对tailwindcss的智能感知提示&#xff0c;在利用tailwindcss实现html效果布局。非常的方便。 生成CSS

Maven POM元素解析(二)

一、parent <parent>元素包含定位此项目将从中继承的父项目所需的信息。注意&#xff1a;此元素的子元素不是插值的&#xff0c;必须作为文字值给定。 ElementTypeDescriptiongroupIdString要从中继承的父项目的组id。artifactIdString要从中继承的父项目的项目id。ver…

python零基础入门笔记【源源老师】

1. print() 输出 &#xff08;1&#xff09;认识英文单词 print: 输出&#xff08;2&#xff09;print() 的作用 print()&#xff1a;用于在控制台上输出你想要的内容。 &#xff08;3&#xff09;代码演示 举例1&#xff1a; 【注意&#xff1a;】用双引号包裹的&#xff0…

java创建线程池的方法

简介 线程池是一种用于管理和重用线程的机制&#xff0c;它可以有效地管理线程的创建和销毁&#xff0c;减少线程创建和销毁的开销&#xff0c;并且能够控制并发线程数量&#xff0c;避免资源耗尽和系统过载。Java 提供了java.util.concurrent 包来支持线程池的实现。 1.Threa…

java(网络编程)

什么是网络编程? 在网络通信协议下&#xff0c;不同计算机上运行的程序&#xff0c;进行的数据传输。 应用场景&#xff1a;即时通信、网游对战、金融证券、国际贸易、邮件、等等 不管是什么场景&#xff0c;都是计算机跟计算机之间通过网络进行数据传输 Java中可以使用ja…

Qt | 事件第二节

Qt | 事件第一节书接上回 四、事件的接受和忽略 1、事件可以被接受或忽略,被接受的事件不会再传递给其他对象,被忽略的事件会被传递给其他对象处理,或者该事件被丢弃(即没有对象处理该事件) 2、使用 QEvent::accept()函数表示接受一个事件,使用 QEvent::ignore()函数表示…