从数组和指针笔试题深度理解 sizeof 和 strlen

2024-06-10 21:20

本文主要是介绍从数组和指针笔试题深度理解 sizeof 和 strlen,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、sizeof 和 strlen 的区别 

a.sizeof

        sizeof 计算变量所占内存内存空间大小的,单位是字节,如果操作数是类型的话,计算的是使用类型创建的变量所占内存空间的大小。sizeof 只关注占用内存空间的大小,不在乎内存中存放什么数据。

b.strlen

        strlen 是C语言库函数,功能是求字符串长度。函数原型如下:

size_t strlen ( const char * str );

        统计的是从 strlen 函数的参数 str 中这个地址开始向后, \0 之前字符串中字符的个数。
strlen 函数会一直向后找 \0 字符,直到找到为止,所以可能存在越界查找。

        具体详见:strlen - C++ Reference (cplusplus.com)

strlen(arr1)的值是未定义的,因为数组arr1中没有 '\0' ,出现越界访问,sizeof(arr2)之所以为4,这是因为字符串的隐含条件 '\0' ,故arr2数组有元素 'a' 、'b'、 'c'、 '\n'。

c.sizeof与strlen的区别

二、数组和指针笔试题解析(32位环境)

a.一维数组( int a[ ] = { 1 , 2 , 3 , 4 } )

1、

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

数组名在sizeof单独出现,其代表的是整个数组,此计算的是整个数组占内存空间的大小,为 4 * 4 = 16字节。

2、

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

数组名不单独在sizeof出现,其代表首元素地址,a + 0 为首元素地址,类型为int* ,指针变量占内存大小为4字节。

3、

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

数组名不单独在sizeof出现,其代表首元素地址,对其解引用,得到数组首元素,类型为int,占内存空间大小为4字节。

4、

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

a[1]访问第一个数组元素,其类型为int,占内存空间大小为4字节。

5、

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

数组名不单独在sizeof出现,其代表首元素地址,对其取地址,那仍然还是一个地址,为二级指针,类型为int**,占内层空间大小为4字节。

6、

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

数组名不单独在sizeof出现,其代表首元素地址,对其取地址又解应用,*&a == a,所以又变成了数组名单独在sizeof出现。

b.字符数组

(char arr[ ] = {'a' , 'b' , 'c' , 'd' , 'e' , 'f' })

1、

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

数组名单独在sizeof内出现代表首元素地址,计算的是整个数组的大小,为 6 * 1 = 6 字节。arr作为首元素地址,意味着strlen从第一个元素开始向后统计,但arr数组中没有 '\0' ,所以统计结果未定义。

2、

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

arr未单独出现,为数组首元素地址,对其解引用得到首元素,strlen库函数在前文中介绍已知,其参数为字符串地址,这里传参错误。

3、

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

arr[0] 为首元素,对其取地址再 +1 ,那么这就是第二个元素的地址喽,其类型为int*,占内存空间大小为4字节,故sizof计算为4字节,但是strlen库函数又是老问题,无论从第几个元素开始,由于没有 '\n' 结尾,结果都是未定义的。

( char arr[ ] = "abcdef" )

1、

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

大多数情况下,数组名都代表首元素地址,除了sizeof数组名或&数组名两个例外,那么这里arr与arr + 0 都代表数组首元素地址,这一次由于字符串末尾隐含 '\0' ,所以可以成功计数,为6。这里的6不是六个字节,而是六个字符元素。

2、

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

数组名单独出现于sizeof,代表整个数组,由于字符串隐含 '\0',所以数组arr实有7个字符元素,占内存空间大小为7 * 1 = 7字节。

3、

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

对数组名取地址,得到的是整个数组的地址,但其值和首元素地址相同,由于没有进形指针运算,那么这里实则还是从数组首元素开始计数,为6字节;对数组首元素取地址得到数组首元素地址从数组首元素开始计数,为6字节。

c.二维数组( int a[3][4] = { 0 } )

1、

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

数组名单独出现于sizeof,代表整个数组,计算的是整个数组大小,占空间 3 * 4 * 4 = 48字节。

2、

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

因为a为二维数组,那么a[0]就代表着第一行首元素的地址,但由于其单独出现于sizeof,计算为第一行元素总大小,为 4 * 4 = 16字节,a[0] + 1并非a[0]单独出现了,就代表第一行首元素大小,那么+1就是第一行第二个元素的地址,但由于其是一个地址,则计算结果为4字节。

3、

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

因为a为二维数组,故数组名代表首元素地址,二二维数组首元素地址为第一行元素数组的地址,但是当其 +1 跳过4 * 4 = 16字节后,仍然还是个地址,作为指针变量,其占内存空间大小就为4字节,对其解引用,其效果相当于第二行元素组成元素的数组名,代表整个第二行数组,计算的是第二行数组大小,其大小为4 * 4 = 16字节。

这篇关于从数组和指针笔试题深度理解 sizeof 和 strlen的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1049298

相关文章

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

JavaScript对象转数组的三种方法实现

《JavaScript对象转数组的三种方法实现》本文介绍了在JavaScript中将对象转换为数组的三种实用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录方法1:使用Object.keys()和Array.map()方法2:使用Object.entr

Rust 智能指针的使用详解

《Rust智能指针的使用详解》Rust智能指针是内存管理核心工具,本文就来详细的介绍一下Rust智能指针(Box、Rc、RefCell、Arc、Mutex、RwLock、Weak)的原理与使用场景,... 目录一、www.chinasem.cnRust 智能指针详解1、Box<T>:堆内存分配2、Rc<T>:

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3