数组和指针笔试题解析之【数组】

news2025/1/11 15:04:08

前言:

1.数组名的意义:

  • sizeof(数组名):这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节。
  • &数组名:这里的数组名表示整个数组,取出的是整个数组的地址。
  • 除此之外所有的数组名都表示首元素的地址。

2.地址在内存中唯一标识一块空间,大小是4/8个字节。在32位平台下(X86环境)是4字节,64位平台下(X64环境)是8字节。

3.sizeof操作符,计算的是对象或者类型创建的对象所占内存空间的大小,单位是字节,它不会关注内存中存放的到底是什么。

4.strlen库函数,求的是字符串长度,本质上是统计字符串中\0之前出现的字符个数。

1.一维数组:

int main()
{
	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));

    return 0;
}

运行结果: 

🎈解析: 

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

数组名a单独放在sizeof内部,数组名表示整个数组,计算的是整个数组的大小,单位是字节,所以是16字节 


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

a并非单独放在sizeof内部,也没有&,所以数组名是数组首元素的地址,即a+0还是首元素的地址,是地址大小就是4/8字节  


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

a并非单独放在sizeof内部,也没有&,所以数组名是数组首元素的地址,*a就是首元素,也可以写成*a == *(a+0) == a[0],大小就是4字节


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

a并非单独放在sizeof内部,也没有&,所以数组名是数组首元素的地址,a+1就是第二个元素的地址,a+1 == &a[1],是第二个元素的地址,是地址大小就是4/8字节 


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

a[1]就是数组的第二个元素,这里计算的就是第二个元素的大小,单位是字节,为4字节 


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

&a是取出数组的地址,但是数组的地址也是地址,是地址就是4/8个字节,数组的地址和数组首元素的地址的本质区别是类型的区别,并非大小的区别

a    ---  int*                      int * p = a;

&a  ---  int (*)[4]            int (*p)[4] = &a; 


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

1.对数组指针解引用访问一个数组的大小,单位是字节,所以是16字节

2.还可以理解为sizeof(*&a) 等价于 sizeof(a)


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

&a是数组的地址,&a+1还是地址,是地址就是4/8个字节 


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

&a[0]就是取出首元素的地址,计算的是地址的大小,就是4/8个字节 


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

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

&a[1] 、&a[0] + 1、a+1都可以取出第二个元素的地址 

2.字符数组 :

2.1题型一:

int main()
{
	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));

	return 0;
}

运行结果: 

🎈解析: 

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

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


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

arr是首元素的地址==&arr[0],是地址就是4/8个字节

注意:

       1. 指针变量的大小和类型无关,不管什么类型的指针变量,大小都是4/8个字节

        2.指针变量是用来存放地址的,地址存放需要多大空间,指针变量的大小就是几个字节

        3.32位环境下,地址是32个二进制位,需要4个字节,所以指针变量的大小就是4个字节

        4.64位环境下,地址是64个二进制位,需要8个字节,所以指针变量的大小就是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/8个字节 


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

&arr[0]是第一个元素的地址,+1就是第二个元素的地址,是地址就是4/8个字节 


🎈下面我们将这个代码稍微变动一下,将sizeof变为strlen,我们再来分析一下 :

注:strlen函数求的是字符串长度,统计的是在字符串中\0之前出现的字符的个数

#include <string.h>
int main()
{
	char arr[] = { 'a','b','c','d','e','f' };//末尾没有\0

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

	return 0;
}

运行结果: 

🎈解析: 

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

arr是首元素的地址,从首元素向后找,找不到\0,所以它的长度就为随机值 


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

 arr是首元素的地址,arr+0还是首元素的地址,所以它的长度也为随机值


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

  arr是首元素的地址,*arr就是首元素-> 'a' -> 97,站在strlen的角度,认为传参进去的 'a' -> 97就是地址,97作为地址,直接进行访问,就是非法访问,所以这是一个错误代码


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

arr[1]就是第二个元素 'b' -> 98,98作为地址,直接进行访问,就是非法访问,所以这也是一个错误代码


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

&arr取出的类型是char (*)[6](数组指针类型),strlen 的类型是const char*,将&arr的类型传给strlen类型必然会进行类型的转换,编译器会报警告,但是传参的时候只是类型发生了变化,值不变,所以还是从第一个元素开始向后找,找不到\0,它的长度也为随机值


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

&arr取出的是整个数组的地址,+1跳过整个数组,向后找还是找不到\0,所以它的长度也为随机值 


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

&arr[0]取出的是第一个元素的地址,+1后指向第二个元素,从第二个元素向后还是找不到\0,所以它的长度也为随机值 


2.2题型二: 

int main()
{
    char arr[] = "abcdef";//a b c d e f \0

    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));

    return 0;
}

运行结果: 

🎈解析: 

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

sizeof(arr)计算的是整个数组的大小,加上\0为7个字节 


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

arr是首元素的地址,加0还是首元素的地址,是地址大小就为4/8个字节 


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

arr是数组名,是首元素的地址,给它解引用就是首元素 'a' ,它的大小就是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/8个字节


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

&arr[0]是第一个元素的地址,加1就是第二个元素的地址,是地址就是4/8个字节 


 🎈下面我们再来将这个代码变动一下,将sizeof变为strlen

#include <string.h>

int main()
{
	char arr[] = "abcdef";
	printf("%d\n", strlen(arr));
	printf("%d\n", strlen(arr + 0));
	//printf("%d\n", strlen(*arr));//err
	//printf("%d\n", strlen(arr[1]));///err
	printf("%d\n", strlen(&arr));
	printf("%d\n", strlen(&arr + 1));
	printf("%d\n", strlen(&arr[0] + 1));
	return 0;
}

运行结果: 

🎈解析: 

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

arr是数组名,即首元素地址,从首元素向后数,统计\0之前的字符,长度为6 


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

 arr是数组名,为首元素地址,加0还是首元素地址,从首元素向后数,统计\0之前的字符,长度也为6 


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

 arr是首元素地址,对首元素地址解引用得到的就是首元素,传给strlen的就是字符'a',即97,所以这是个错误代码


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

arr[1]是这个数组的第二个元素,即把字符'b'传了进去,所以它也是个错误代码 


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

&arr取出的是整个数组的地址,但它也是从a向后数,直到\0截至,所以它的长度为6 


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

&arr+1是跳过整个数组,包括后边的\0,从\0后边开始数,不确定什么时候会遇到\0,所以它的长度为随机值 


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

&arr[0]+1得到的是第二个元素的地址,从第二个元素开始向后数,得到的长度就是5 


2.3题型三: 

int main()
{
    //把字符串首字符的地址放到了p里边去,也就是说p指向了"abcdef"
	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));
	printf("%d\n", sizeof(&p + 1));
	printf("%d\n", sizeof(&p[0] + 1));
	return 0;
}

运行结果: 

🎈解析: 

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

 p是一个指针变量,计算的就是指针变量的大小,就是4/8字节


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

p是一个char*的指针,+1向后偏移一个字节,但它本质上还是一个地址,大小是4/8个字节 


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

 p是char*的指针,解引用访问一个字节,所以它的大小就是1字节


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

p[0] ---> *(p+0) ---> *p == 'a',所以它的大小也是1字节


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

&p也是地址(指针变量p的起始地址),大小也是4/8个字节 


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

&p是地址,&p+1还是地址,是地址就是4/8个字节 


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

&p[0]是第一个元素的地址,&p[0]+1就是第二个元素的地址,大小就是4/8个字节 


🎈 将sizeof变为strlen,我们继续分析:

int main()
{
	char* p = "abcdef";//a b c d e f \0

	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));
	return 0;
}

🎈解析: 

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

 p指向了字符串中的首字符'a',从a向后数,截至到\0,长度就为6


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

p+1跳过一个字节,从第二个元素向后数,长度就是5 


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

p指向字符串中第一个元素的位置,解引用就是'a',所以这是个错误代码 


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

p[0]也是第一个元素,传给strlen的也是'a'的ASCII码值,错误代码 


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

 &p指向了指针变量p的起始地址,在这个内存里边有没有\0以及什么时候遇到\0完全是不可知的,所以它的长度就是一个随机值


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

&p+1是跳过p,指向的下一块内存后续空间有什么也是不可知的,所以它的长度也是随机值 


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

 &p[0]+1就是第二个元素的地址,从第二个元素向后数长度就是5

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

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

相关文章

爬虫逆向实战(33)-某联社数据(webpack)

一、数据接口分析 主页地址&#xff1a;某联社 1、抓包 通过抓包可以发现数据接口是/nodeapi/telegraphList 2、判断是否有加密参数 请求参数是否加密&#xff1f; 通过查看“载荷”模块可以发现有一个sign加密参数 请求头是否加密&#xff1f; 无 响应是否加密&#x…

uni-app 新增 微信小程序之新版隐私协议

一、manifest.json中配置 "__usePrivacyCheck__": true 二、编写封装后的组件 <template><view class"privacy" v-if"showPrivacy"><view class"content"><view class"title">隐私保护指引</…

无涯教程-JavaScript - SHEET函数

描述 SHEET函数返回参考图纸的图纸编号。 语法 SHEET (value)争论 Argument描述Required/OptionalValue 值是您要为其指定工作表编号的工作表名称或参考。 如果省略value,则SHEET返回包含该功能的工作表的编号。 Optional Notes 除了所有其他工作表类型(宏,图表或对话框工…

【Python爬虫】python打印本地代理

目录 前言 代理 IP 的使用 1. 获取代理 IP 2. 选择合适的代理 IP 3. 设置代理 IP 4. 验证代理 IP 代码案例 总结 前言 在进行网络爬虫时&#xff0c;使用代理是非常重要的。因为爬虫经常会被网站封 IP&#xff0c;而代理可以隐藏你的真实 IP 地址&#xff0c;让你可以…

智慧港口4G+UWB+GPS/北斗RTK人员定位系统解决方案

港口人员定位系统能够帮助企业实现对港口作业人员的全面监控和管理&#xff0c;不仅可以保障人员的人身安全&#xff0c;还可以提高人员的作业效率&#xff0c;为港口的可持续发展提供有力保障。接下来为大家分享智慧港口人员定位系统解决方案。 方案背景 1、港口作业人员多&a…

APK安装过程解析

应用端发起安装APK的代码一般如下&#xff1a; Intent installintent new Intent();installintent.setAction(Intent.ACTION_VIEW);installintent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);installintent.setDataAndType(xxx,"application/vnd.android.package-archive&…

SpringMVC系列(四)之SpringMVC实现文件上传和下载

目录 前言 一. SpringMVC文件上传 1. 配置多功能视图解析器 2. 前端代码中&#xff0c;将表单标记为多功能表单 3. 后端利用MultipartFile 接口&#xff0c;接收前端传递到后台的文件 4. 文件上传示例 1. 相关依赖&#xff1a; 2. 逆向生成对应的类 3. 后端代码&#xf…

DC电源模块在保护设备损坏的重要功能

BOSHIDA DC电源模块在保护设备损坏的重要功能 DC电源模块是一种电源管理设备&#xff0c;用于将交流电转换为直流电并提供给设备供电。它通常由多个电子元件组成&#xff0c;包括整流器、滤波器、稳压器等&#xff0c;以确保电源输出稳定&#xff0c;满足设备的电源需求。 在…

“文件管理技巧:批量归类相同名称的文件到指定文件夹“

在日常生活和工作中&#xff0c;我们经常需要处理大量的文件&#xff0c;如果每个文件都单独归类整理&#xff0c;会浪费大量的时间和精力。有没有一种简单的方法可以批量将相同名称的文件归类到指定文件夹里呢&#xff1f;答案是肯定的&#xff01;下面就让我们一起来了解这个…

JavaScript对象实战及应用

&#x1f3ac; 岸边的风&#xff1a;个人主页 &#x1f525; 个人专栏 :《 VUE 》 《 javaScript 》 ⛺️ 生活的理想&#xff0c;就是为了理想的生活 ! 目录 引言 1. 对象属性 访问属性 修改属性 删除属性 动态添加属性 属性枚举 属性描述符 2. 对象 API Object.ke…

cms之wordpress安装教程

1、下载程序 到wordpress官方网站下载wordpress程序&#xff0c;官方下载地址&#xff1a;https://cn.wordpress.org/download/。 下载最新版的wordpress程序 https://cn.wordpress.org/latest-zh_CN.zip 2、上传程序 上传程序前先确认主机是否符合安装的环境要求&#xff…

芯科蓝牙BG27开发笔记7-配置蓝牙参数

基础的要求 1. 设置广播参数为间隔1000ms&#xff0c;不停止 2. 添加广播消息&#xff0c;含01、03、09、FF TYPE 3. 设置蓝牙通信间隔参数为320ms、400ms、2、4000ms超时 3. 配置发射功率为较低 4. 配置GATT所有数据与原Nordic 配置一致 为了解决以上疑问&#xff0c;需…

4.zigbee开发,传感器网络管理进阶(网状和树状拓扑),zigbee的ADC

一。zigbee的串口 1.串口通信的基本概念 &#xff08;1&#xff09;同步通信与异步通信 同步通信&#xff1a; 一般情况下同步通信指的是通信双方根据同步信号进行通信的方式。比如通信双方有一个共同的时钟信号&#xff0c;通讯中通常双方会统一规定在时钟信号的上升沿…

DP专题3 使用最小花费爬楼梯

题目&#xff1a; 思路&#xff1a; 根据题意&#xff0c;我们先明确 dp 数组 i 的含义&#xff0c; 这里很明显&#xff0c;可以知道 i 是对应阶梯的最少花费&#xff0c; 其次dp初始化中&#xff0c;我们的 dp[0] 和 dp[1] 是 0 花费&#xff0c; 这是我们可以选择的&am…

关键词生成原创文章软件-原创文章生成软件

大家好&#xff0c;今天我想和大家分享一下我对147SEO关键词生成原创文章工具的感受。作为一个经常需要写作的人&#xff0c;我深知寻找创意和构建文章结构的挑战。关键词生成原创文章似乎为这些问题提供了一种解决方案。 首先&#xff0c;让我谈谈我的感受。关键词生成原创文章…

9个值得收藏的WebGL性能优化技巧

在这里&#xff0c;我们推荐一些经证明非常适合创建基于 Web 的交互体验的优化技术。 本章主要基于 Soft8Soft 在 Verge3Day Europe 2019 会议上的演讲。 推荐&#xff1a;用 NSDT编辑器 快速搭建可编程3D场景 1、几何/网格 几何是 3D 应用程序的基础&#xff0c;因为它构成了…

华为云云耀云服务器实例使用教学

目录 国内免费云服务器&#xff08;体验&#xff09; 认识国内免费云服务器 如何开通国内免费云服务器 云耀云服务器 HECS Xshell 远程连接 云服务器更改安全组 切换操作系统 服务器详情 HECS适用于哪些场景&#xff1f; 网站搭建 电商建设 开发测试环境 云端学习环…

二维码智慧门牌管理系统开发解决方案:标准化建设的基础

文章目录 前言一、系统质量保证二、系统互联互通三、系统可扩展性 前言 在现代城市管理和服务中&#xff0c;二维码智慧门牌管理系统扮演着至关重要的角色&#xff0c;它通过智能化和数字化手段提高了城市管理效率、公共服务水平&#xff0c;并有助于维护社会公共安全。然而&a…

macOS 12 Monterey:一次全新的跨设备协作体验

macOS 12 Monterey是苹果公司的一次重大突破&#xff0c;它打破了设备间的壁垒&#xff0c;将不同设备无缝地连接在一起&#xff0c;极大地提升了用户的工作效率和娱乐体验。Monterey带来了通用控制、AirPlay、捷径等新功能&#xff0c;以及一些实用的新小功能。 安装&#xf…

跨链协议支持Sui的资产所有权理念,助力资产在不同链之间流通

区块链通常支持安全地持有数字资产这一概念。然而&#xff0c;在一个链上拥有资产并不意味着它可以转移到另一个链上。支持在不同链之间移动资产的跨链协议有助于解决行业中可能出现的主要碎片化问题。 Sui通过基于开源Wormhole协议构建的Wormhole Connect支持跨链。构建者可以…