C语言 --- 指针(5)

news2024/10/1 21:33:04

目录

一.sizeof和strlen对比

1.sizeof

2.strlen

3.strlen 和sizeof的对比

二.数组和指针笔试题目详解

 回顾:数组名的理解

1.一维数组

2.字符数组 

代码1:

 代码2:

 代码3:

代码4:

代码5:

代码6:

3.二维数组 

总结 


一.sizeof和strlen对比

1.sizeof

sizeof既是一种关键字也是一个操作符。sizeof的作用是计算变量和类型的所占内存空间的大小,单位是字节,返回size_t类型的值。sizeof只关注所占用内存空间的大小,不在乎内存中存放的什么数据。

比如:

#include<stdio.h>
int main()
{
    int a = 10;
    printf("%d\n",sizeof(a));
    printf("%d\n",sizeof a );
    printf("%d\n",sizeof(int));
    
    return 0;
}

sizeof在使用时,可以不用括号,这证明了它不是一个函数,但是在计算类型比如int就不行,必须加括号。

打印结果都是4,size_t类型的值应该用%zd打印,但是这里也可以使用%d,只不过编译器会发出警告,但是影响不大。

sizeof不关注括号中的数据,只关注存放内存的大小。比如:

#include<stdio.h>
int main()
{
	int a = 10;
	int size = sizeof(a++);
	printf("%d\n",size);
	printf("%d\n",a);
	return 0;
}

输出结果是4  10

很明显sizeof括号中的表达式是没有计算的。 

2.strlen

strlen是C语言库函数,在使用时需要包含头文件<string.h>

函数原型:

size_t strlen(const char *str);

这个函数会返回字符串中字符数,但是不包括\0,也就是返回的是\0之前的字符数。如果没有\0就会出现越界查找的情况。

比如:

#include<stdio.h>
int main()
{
    char arr1[] = {'a','b','c'};
    char arr2[] ="abc" ;
    printf("%d\n",strlen(arr1));
    printf("%d\n",strlen(arr2));
    printf("%d\n",sizeof(arr1));
    printf("%d\n",sizeof(arr2));
    
    return 0;
}

打印结果是 随机值 3 3 4.

第一个是随机值的原因就是没有\0,strlen会一直查找直到遇到\0才终止。

第二个因为是常量字符串,包含\0,所以是3。

第三个计算的是数组arr1中所占内存的大小,只有三个字符,也就是3个字节。

第四个arr2中有\0,所以是4个字节。

3.strlen 和sizeof的对比

sizeofstrlen
  1. sizeof是操作符。
  2. sizeof计算的是操作数所占内存空间的大小,单位是字节。
  3. 不关注内存中存放的什么数据。
  1. strlen是库函数,使用的头文件是<string.h>。
  2. strlen是求字符串长度的,统计的是\0之前的字符个数。
  3. strlen关注内存中是否有\0,如果没有\0,就会继续往前找,可能会出现越界的情况。

二.数组和指针笔试题目详解

 回顾:数组名的理解

 数组名是数组首元素的地址。

但是有两个的特殊情况

  1. sizeof(数组名) ,sizeof中单独放数组名,这里的数组名表示的是整个数组,计算的是整个数组的大小,单位是字节。
  2. &数组名,这里的数组名代表的是整个数组,&取出的是整个数组的地址。(整个数组的地址虽然和数组首元素的地址是相同的,但是指针变量的类型是不同的。)

1.一维数组

#include<stdio.h>
int main()
{
	int a[] = { 1,2,3,4 };

	printf("%d\n", sizeof(a));//1
	printf("%d\n", sizeof(a + 0));//2
	printf("%d\n", sizeof(*a));//3
	printf("%d\n", sizeof(a + 1));//4
	printf("%d\n", sizeof(a[1]));//5
	printf("%d\n", sizeof(&a));//6
	printf("%d\n", sizeof(*&a));//7
	printf("%d\n", sizeof(&a + 1));//8
	printf("%d\n", sizeof(&a[0]));//9
	printf("%d\n", sizeof(&a[0] + 1));//10

	return 0;
}
序号解析答案
1 a是数组名,sizeof中单独放数组名,计算的是整个数组的大小,单位字节。16
2sizeof中没有单独放数组名a,所以这里的数组名就是数组首元素的地址,地址就是指针变量。在不同环境下大小不同。4 || 8
3a是数组首元素的地址,解引用后得到数组首元素。所以计算的是整形变量的大小。4
4a表示数组首元素的地址,地址就是指针,类型是int*,指针加1跳过4个字节,也就是一个整形元素,也就是第二个元素的地址。4 || 8
5a[1]表示数组第二个元素4
6&a是整个数组的地址,地址就是指针变量.4 || 8
7&a是整个数组的地址,解引用后就是整个数组。所以sizeof计算的是整个数组的大小(&和*可以相互抵消16
8&a是整个数组的地址,+1跳过了整个数组,虽然这个地址并没有指向这个数组了,但是我们并没有越界访问,所以不影响sizeof计算这个指针变量的大小。4 || 8
9&a[0]得到的是数组首元素的地址.4 || 8
10&a[0]+1,跳过一个元素,得到的是数组第二个元素的地址。4 || 8

2.字符数组 

代码1:

char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));//1
printf("%d\n", sizeof(arr+0));//2
printf("%d\n", sizeof(*arr));//3
printf("%d\n", sizeof(arr[1]));//4
printf("%d\n", sizeof(&arr));//5
printf("%d\n", sizeof(&arr+1));//6
printf("%d\n", sizeof(&arr[0]+1));//7
序号解析答案
1arr单独放在sizeof中,计算的是整个数组的大小。单位字节6
2arr没有单独放在数组中,arr表示的是数组首元素的地址4 || 8
3arr表示的是数组首元素的地址,解引用后得到的是数组首元素。1
4arr[1]表示数组首元素。1
5&arr表示整个数组的地址,地址就是指针变量4 || 8
6

&a是整个数组的地址,+1跳过了整个数组,虽然这个地址并没有指向这个数组了,但是我们并没有越界访问,所以不影响sizeof计算这个指针变量的大小。

4 || 8
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址4 || 8

 代码2:

char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", strlen(arr));//1
printf("%d\n", strlen(arr+0));//2
printf("%d\n", strlen(*arr));//3
printf("%d\n", strlen(arr[1]));//4
printf("%d\n", strlen(&arr));//5
printf("%d\n", strlen(&arr+1));//6
printf("%d\n", strlen(&arr[0]+1));//7
序号解析答案
1arr是数组首元素的地址,strlen计算的是'\0'之前的字符个数,但是这个数组中并没有'\0',所以会出现越界的情况,故无法出现准确的值的情况。随机值
2这里的arr也是数组首元素的地址,+0并没有跳过任何字节,与序号1同理也是不确定情况。(两个值应该是一样的)随机值
3strlen的参数是const char *类型的。*arr得到的是数组首元素,将这个作为参数传给strlen是不行的,这里程序会崩溃。error
4arr[1]表示数组第二个元素,将其作为参数传给strlen是不行的。error
5&arr表示整个数组的地址,虽然是一个数组指针变量,但是传参会进行强制类型转换为char*类型,值和数组首元素的地址相同,所以和序号1和2相同,值一样的。随机值
6

&a是整个数组的地址,+1跳过了整个数组,也就是跳过了6个字节。但是因为没有'\0',也会出现随机的情况

随机值(比序号125小6)
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,只跳过了1个字节。随机值(比序号125小1)

输出结果:(以下是随机值,不是特定情况。)

 

序号3和4已被注释掉了。

 代码3:

char arr[] = "abcdef";
printf("%d\n", sizeof(arr));//1
printf("%d\n", sizeof(arr+0));//2
printf("%d\n", sizeof(*arr));//3
printf("%d\n", sizeof(arr[1]));//4
printf("%d\n", sizeof(&arr));//5
printf("%d\n", sizeof(&arr+1));//6
printf("%d\n", sizeof(&arr[0]+1));//7
序号解析答案
1sizeof(数组名),计算的是整个数组的大小,字符串末尾是有一个'\0'的,只不过没有显示出来,我们可以通过调试观察,所以这个字符串中有7个字符。7
2这里的arr也是数组首元素的地址,+0并没有跳过任何字节,是一个指针变量。4 || 8
3*arr的到的数组首元素。1
4arr[1]表示数组第二个元素.1
5&arr表示整个数组的地址,是一个指针变量。4 || 8
6

&a是整个数组的地址,+1跳过了整个数组,但是还是一个指针变量

4 || 8
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,是一个指针变量4 || 8

序号1:

我们可以看到数组中分别放着不同的字符,字符前面是它们的ASCII码值,arr[6],也就是数组的第7个元素,存放着'\0'字符。

代码4:

char arr[] = "abcdef";
printf("%d\n", strlen(arr));//1
printf("%d\n", strlen(arr+0));//2
printf("%d\n", strlen(*arr));//3
printf("%d\n", strlen(arr[1]));//4
printf("%d\n", strlen(&arr));//5
printf("%d\n", strlen(&arr+1));//6
printf("%d\n", strlen(&arr[0]+1));//7
序号解析答案
1sizeof计算的是'\0'之前的地址,有6个字符6
2sizeof计算的是'\0'之前的地址,有6个字符6
3*arr的到的数组首元素。不应该作为参数传给strlenerror
4arr[1]表示数组第二个元素.不应该作为参数传给strlenerror
5&arr表示整个数组的地址,sizeof计算的是'\0'之前的地址,有6个字符.6
6

&a是整个数组的地址,+1跳过了整个数组,故不知道什么时候会遇到'\0',所以是个随机值

随机值
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,从第二个开始到'\0',有5个字节。5

输出结果:

序号6就是经过了16个字符才遇到'\0',error情况已经注释掉了。

代码5:

char *p = "abcdef";
printf("%d\n", sizeof(p));//1
printf("%d\n", sizeof(p+1));//2
printf("%d\n", sizeof(*p));//3
printf("%d\n", sizeof(p[0]));//4
printf("%d\n", sizeof(&p));//5
printf("%d\n", sizeof(&p+1));//6
printf("%d\n", sizeof(&p[0]+1);//7
序号解析答案
1sizeof计算的是'\0'之前的地址,有6个字符6
2sizeof计算的是'\0'之前的地址,有6个字符6
3*arr的到的数组首元素。不应该作为参数传给strlenerror
4arr[1]表示数组第二个元素.不应该作为参数传给strlenerror
5&arr表示整个数组的地址,sizeof计算的是'\0'之前的地址,有6个字符.6
6

&a是整个数组的地址,+1跳过了整个数组,故不知道什么时候会遇到'\0',所以是个随机值

随机值
7&arr[0]表示数组首元素的地址,+1跳过一个元素,表示数组第二个元素的地址,从第二个开始到'\0',有5个字节。5

代码6:

char *p = "abcdef";
printf("%d\n", strlen(p));//1
printf("%d\n", strlen(p+1));//2
printf("%d\n", strlen(*p));//3
printf("%d\n", strlen(p[0]));//4
printf("%d\n", strlen(&p));//5
printf("%d\n", strlen(&p+1));//6
printf("%d\n", strlen(&p[0]+1));//7
序号解析答案
1sizeof计算的是'\0'之前的地址,有6个字符6
2sizeof计算的是'\0'之前的地址,+1后跳过一个字节,也就是一个元素,有5个字符5
3*p的到的数组首元素。不应该作为参数传给strlenerror
4p[0]表示数组第一个元素.不应该作为参数传给strlenerror
5&p表示字符指针的地址,sizeof计算的是'\0'之前的地址,会产生随机值随机值
6

&p是字符指针的地址,加1 跳过4 || 8个字节,故不知道什么时候会遇到'\0',所以是个随机值

随机值
7&p[0]表示首首字符的地址,+1跳过一个元素,表示第二个字符的地址,从第二个开始到'\0',有5个字节。5

输出结果:

3.二维数组 

int a[3][4] = {0};
printf("%d\n",sizeof(a));//1
printf("%d\n",sizeof(a[0][0]));//2
printf("%d\n",sizeof(a[0]));//3
printf("%d\n",sizeof(a[0]+1));//4
printf("%d\n",sizeof(*(a[0]+1)));//5
printf("%d\n",sizeof(a+1));//6
printf("%d\n",sizeof(*(a+1)));//7
printf("%d\n",sizeof(&a[0]+1));//8
printf("%d\n",sizeof(*(&a[0]+1)));//9
printf("%d\n",sizeof(*a));//10
printf("%d\n",sizeof(a[3]));//11
序号解析

答案

1数组名a单独放在sizeof中,计算的是整个数组的大小。48 = 3 * 4 * sizeof(int)
2a[0][0]表示数组第一行第一个元素。4
3
  1. a[0]表示二维数组首元素地址,也就是第一个一维数组的数组名,a[0]就是单独存放在sizeof的数组名,计算的就是第一个一维数组的大小.
  2. a[0]可以写成*(a + 0),a是二维数组的第一个一维数组的地址,解引用就是第一个一维数组了.
4 * sizeof(int) = 16
4a[0]表示第一个一维数组的数组名,并没有单独放在sizeof中,代表首元素的地址,+1,跳过一个元素,也就是&a[0][1].4 || 8
5解引用后得到的是a[0][1]4
6a表示数组首元素的地址,+1就是第二个元素的地址,也就是第二个一维数组的地址,4 || 8
7解引用后就是第二个一维数组。16
8a[0]是第一行的数组名,&a[0]就是第一行的地址,+1后是第二行的地址,地址就是指针4 || 8
9解引用后就是第二行,计算的第二行的大小16
10*a就是第一行,计算的就是第一行的大小16
11a[3]是第四行的数组名,虽然没有第四行,但是sizeof并没有进行访问,所以并不是越界访问,根据类型仍然可以计算第四行的大小,4个整形变量。16

总结 

数组名的意义:

  1.  sizeof(数组名),这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩。
  2.  & 数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址。
  3. 除此之外所有的数组名都表⽰⾸元素的地址。

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

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

相关文章

《JAVA与模式》之门面模式

系列文章目录 文章目录 系列文章目录前言一、医院的例子二、门面模式的结构三、门面模式的实现四、门面模式的优点五、门面模式在Tomcat中的使用前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站,这篇文章男女通用,…

【论文阅读】OpsEval

粗糙翻译&#xff0c;如有兴趣请看原文&#xff0c;链接&#xff1a;https://arxiv.org/abs/2310.07637 摘要 信息技术&#xff08;IT&#xff09;运营&#xff08;Ops&#xff09;&#xff0c;特别是用于IT运营的人工智能&#xff08;AlOps&#xff09;&#xff0c;是保持现…

改进沙猫群优化的BP神经网络ISCSO-BP(时序预测)的Matlab实现

改进沙猫群优化的BP神经网络&#xff08;ISCSO-BP&#xff09;是一种结合了改进的沙猫群优化算法&#xff08;Improved Sand Cat Swarm Optimization, ISCSO&#xff09;和反向传播&#xff08;Back Propagation, BP&#xff09;神经网络的模型&#xff0c;旨在提高时序预测的准…

LLM 推理优化探微 (2) :Transformer 模型 KV 缓存技术详解

编者按&#xff1a;随着 LLM 赋能越来越多需要实时决策和响应的应用场景&#xff0c;以及用户体验不佳、成本过高、资源受限等问题的出现&#xff0c;大模型高效推理已成为一个重要的研究课题。为此&#xff0c;Baihai IDP 推出 Pierre Lienhart 的系列文章&#xff0c;从多个维…

王道机试C++第 5 章 数据结构一:向量vector和蓝桥杯13年两个程序 Day31

5.1 向量 有限个类型相同的变量的线性集合&#xff0c;组成数组的各个变量称为数组的元素。给每个元素分配唯一的一个下标&#xff0c;就能用这个下标指代该元素。还可通过下标直接访问数组中的任何一个元素&#xff0c;并且这些访问能在常数时间内完成。 1&#xff0e;STL-v…

算法练习:同向双指针(滑动窗口)

目录 1. 长度最小的数组2. 无重复字符的最长字符串3. 最大连续1的个数4. 将x减到0的最小操作数5. 水果成篮6. 找到字符串中所有字母异位词7. 串联所有单词的子串8. 最小覆盖子串 1. 长度最小的数组 题目信息&#xff1a; 题目链接&#xff1a; 长度最小的数组思路&#xff1a; …

观察主线程退出对子线程的影响

✨主进程以pthread_exit(0)方式退出——线程退出 ✨子线程pthread_cancel方式杀死父线程 ✨主线程以return&#xff0c;exit(0)方式退出——进程退出  &#x1f531;return   &#x1f531;return在主函数   &#x1f536;return在子函数  &#x1f536;exit(0)   &…

数据结构——堆的应用 Topk问题

&#x1f49e;&#x1f49e; 前言 hello hello~ &#xff0c;这里是大耳朵土土垚~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#x1f339;&#x1f339;&#x1f339; &#x1f4a5;个人主页&#x…

谷歌开源的LLM大模型 Gemma 简介

相关链接&#xff1a; Hugging face模型下载地址&#xff1a;https://huggingface.co/google/gemma-7bGithub地址&#xff1a;https://github.com/google/gemma_pytorch论文地址&#xff1a;https://storage.googleapis.com/deepmind-media/gemma/gemma-report.pdf官方博客&…

多址接入技术 ---FDMA、TDMA、CDMA

1、多址接入技术 所谓多址技术就是把同一个无线信道按照时间、 频率等进行分割, 使不同的用户都能够在不同的分割段中使用这一信道, 而又不会明显地感觉到他人的存在, 就好像自己在专用这一信道一样。 占用不同的分割段就像是拥有了不同的地址, 使用同一信道的多个用户就…

Vue.js过滤器:让数据展示更灵活

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

Unity的相机跟随和第三人称视角二

Unity的相机跟随和第三人称视角二 展示介绍第二种相机模式远离代码总结 展示 我录的视频上可能看不太出来&#xff0c;但是其实今天的这个方法比原来写的那个方法更简便并且死角更少一些。 这里我也做了两个人物偏移的视角&#xff0c;按Q是原来的两个相机模式切换&#xff0c…

问题 | 程序员求职宝典

程序员的金三银四求职宝典 随着春天的脚步渐近,对于许多程序员来说,一年中最繁忙、最重要的面试季节也随之而来。金三银四,即三月和四月,被广大程序员视为求职的黄金时期。在这两个月里,各大公司纷纷开放招聘,求职者们则通过一轮又一轮的面试,力争心仪的职位。而如何在…

从零学习Linux操作系统 第三十二部分 Ansible中的变量及加密

一、变量的基本设定及使用方式 变量命名&#xff08;与shell相同&#xff09;&#xff1a; 只能包含数字&#xff0c;下划线&#xff0c;字母 只能用下划线或字母开头 .变量级别&#xff1a; 全局从命令行或配置文件中设定的paly:在play和相关结构中设定的主机:由清单&…

NeRF模型NeRF模型

参考视频&#xff1a;https://www.youtube.com/watch?vHfJpQCBTqZs&ab_channelVision%26GraphicsSeminaratMIT NeRF模型的输入输出: 输入: (x, y, z): 一个三维空间坐标,代表场景中的一个位置点(θ, φ): 视线方向,θ表示与y轴的夹角,φ表示与x轴的夹角,用两个角度可以…

(C语言)strcpy与strcpy详解,与模拟实现

目录 1. strcpy strcpy模拟实现&#xff1a; 实现方法1&#xff1a; 实现方法2&#xff1a; 2. strcat strcat模拟实现&#xff1a; 1. strcpy 作用&#xff1a;完成字符串的复制。 头文件&#xff1a;<string.h> destination是字符串要复制到的地点&#xff0c;s…

图片在div完全显示

效果图&#xff1a; html代码&#xff1a; <div class"container" style" display: flex;width: 550px;height: 180px;"><div class"box" style" color: red; background-color:blue; width: 50%;"></div><div …

FreeRTOS操作系统学习——同步互斥与通信

同步&#xff08;Synchronization&#xff09; 同步是一种机制&#xff0c;用于确保多个任务能够按照特定的顺序协调执行或共享数据。当一个任务需要等待其他任务完成某个操作或满足某个条件时&#xff0c;同步机制可以帮助任务进行协调和等待。 在FreeRTOS中&#xff0c;常见…

【布局:1688,阿里海外的新筹码?】1688重新布局跨境海外市场:第一步开放1688API数据采集接口

2023年底&#xff0c;阿里巴巴“古早”业务1688突然成为“重头戏”&#xff0c;尤其宣布正式布局跨境业务的消息&#xff0c;一度引发电商圈讨论。1688重新布局跨境海外市场&#xff1a;第一步开放1688API数据采集接口 2023年11月中旬&#xff0c;阿里财报分析师电话会上&…

SpringBoot项目如何部署到服务器

文章目录 准备&#xff1a;方式一&#xff1a;Jar包方式&#xff08;推荐&#xff09;部署步骤&#xff1a; 方式二&#xff1a;War包方式部署步骤&#xff1a; 总结 准备&#xff1a; 云服务器&#xff08;阿里云、腾讯云等&#xff09;Linux系统以及运行所需的环境 方式一&a…