C/C++指针和数组笔试题解析

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: C/C++指针和数组笔试题解析

1.前言

       在前面,我们学习了指针基础和指针进阶的部分,想再次学习的可以点击指针进阶指针进阶指针初阶。这一次主要和大家一起学习指针的笔试题,这写题非常有意思,用到sizeof函数和strlen函数,接下来就让我们感受一下这些题目的乐趣吧。我们在做题需要知道

  • sizeof(数组名)这里是计算整个数组的大小,也就是所占的字节数。
  • &数组名得到的是整个数组的地址
  • 其余的数组名都是首元素的地址。

2.一维数组

2.1习题一

#include<stdio.h>
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;
}

       在 sizeof(a)中,a是数组名计算的是数组的大小,故为16个字节;在sizeof(a+0).数组名a没有单独存在,因此是首元素地址,既然是地址,那么就占用4/8个字节,sizeof(*a),数组名没有单独存在,*a为首元素,占4个字节,sizeof(a+1),数组名没有单独存在,a为首元素地址,a+1为第二个元素地址,是地址,占用4/8个字节,sizeof(a[1]),是第二个元素,元素为int占4个字节,sizeof(&a),是数组的地址,属于地址,占4/8个字节,sizeof(*&a),我们知道*和&放一起可以抵消,因此相当于sizeof(a),故占16个字节,sizeof(&a+1),是&a是数组的地址,加1是加16个字节,但是他是地址,故占4/8个字节,sizeof(&a[0]),是首元素的地址,占4/8个字节,sizeof(&a[0]+1)是第二个元素的地址,占4/8个字节。

2.2习题二

#include<stdio.h>
#include <string.h>
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));
  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));
  return 0;
}

        sizeof(arr),中arr单独存在,是,故为6个字节,sizeof(arr+0),由于数组名没有单独存在,故属于首元素地址,属于地址,占用4/8个字节,sizeof(*arr),数组名没有单独存在,*arr是首元素,首元素是字符类型,占用1个字节,sizeof(arr[1]),是第二个元素,占用1个字节,sizeof(&arr),是数组的地址,属于地址,占用4/8个字节,sizeof(&arr+1),&arr是数组的地址,&arr+1是跳过这个数组,但还是地址,属于地址,占用4/8个字节,sizeof(&arr[0]+1),&arr[0]是首元素地址,&arr[0]+1是第二个元素的地址,属于地址,占用4/8个字节。对于strlen(arr),数组名是单独存在,但是\0不知道什么时候出现,故为随机值,strlen(arr+0),得到的是首地址,但是不知道\0=什么时候出现,故为随机值,strlen(*a)和strlen(arr[1])中*a和arr[1]都是字符,利用ASCLL进行转换会进行整形提升补零,由于电脑会将它看为地址,但是由于这块地址没有进行初始化,也就是说是野指针,这就会导致电脑出错,程序崩溃,对于最后三局都是因为没有\0所以都是随机值。

2.3习题三

#include<stdio.h>
#include <string.h>
int main()
{
  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));
  return 0;
}

       对于sizeof(arr),数组名单独存在,故为6个字节,sizeof(arr+0),数组名没有单独存在,是首元素的地址,属于地址,占4/8个字节,sizeof(*arr),数组名没有单独存在,是首元素,首元素是char类型,占1个字节。sizeof(arr[1]),是数组的第二个元素,是char类型占1个字节。sizeof(&arr)和sizeof(&arr+1),&arr是数组的地址,&arr+1是数组后面的地址,属于地址,占4/8个字节,sizeof(&arr[0]+1),&arr[0]是首元素的地址,&arr[0]+1是第二个元素的地址,属于地址占4/8个字节。对于strlen(arr),数组名单独存在,故为6,strlen(arr+0),arr没有单独存在,属于首元素地址,故为6个字节。strlen(*a)和strlen(arr[1])中*a和arr[1]都是字符,利用ASCLL进行转换会进行整形提升补零,由于电脑会将它看为地址,但是由于这块地址没有进行初始化,也就是说是野指针,这就会导致电脑出错,程序崩溃。对于后面三个,&arr得到的是数组的地址,&arr[0]+1得到的是第二个元素的地址,&arr+1得到的是数组后面的地址会得到随即值。

3.二维数组

#include<stdio.h>
#include <string.h>
int main()
{
  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]));
  return 0;
}

       对于sizeof(a),数组名单独存在,是数组的地址,故为4*3*4=48个字节,对于sizeof(a[0]),这里比较特殊首先它是单独存在的,我们可以将二维数组看成一维数组的数组,a[0]也就是第一行的地址,故为4/8个字节,对于sizeof(a[0]+1),a[0]是一维数组的数组名,由于a[0]没有单独在sizeof()里故是一维数组的首元素地址,占4/8个字节,那么sizeof(*(a[0]+1))也就是第二个元素,占4个字节,a是二维数组的数组名,sizeof(a+1)中a没有单独存在,故是二维数组的首元素,也就是第一行的地址,a+1也就是第二行的地址,占4/8个字节(我们对于二维数组,可以将二维数组看成一维数组的数组,例如arr[3][4],二维数组的数组名为arr,一维数组的数组名为arr[3]),对于sizeof(*(a+1)),a是二维数组的数组名,没有单独存在,故是二维数组的首元素,也就是第一行的地址,a+1就是第二行的地址*(a+1)就是第二行所有元素,占16个字节。sizeof(&a[0]+1),a[0]是一维数组的数组名,&a[0]是取地址数组名是第一行的地址,&a[0]+1就是第二行的地址,属于地址占4/8个字节,对它进行*解引用占16个字节,对于sizeof(a[3]),虽然数组已经溢出,但是,他还是16个字节,因为函数不会真正进入,它就和a[0]一样。

4.总结

       看完这些题,我可以总结为;

  • sizeof(数组名),其中数组名是整个数组。
  • &数组名加减整数其中&数组名是整个数组的地址,。即使在sizeof()中也是整个数组的地址
  • 对于二维数组,我们可以将二维数组看成一维数组的数组,例如arr[3][4],二维数组的数组名为arr,一维数组的数组名为arr[3],&arr ,arr都是二维数组的地址,但是在sizeof中arr加整数表示第几行的地址,arr[整数] &arr[整数]都表示第几行的地址。

今天的内容就结束了,希望大家可以一键三连。

 

目录
相关文章
|
23天前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
36 5
|
2月前
|
存储 算法 安全
基于红黑树的局域网上网行为控制C++ 算法解析
在当今网络环境中,局域网上网行为控制对企业和学校至关重要。本文探讨了一种基于红黑树数据结构的高效算法,用于管理用户的上网行为,如IP地址、上网时长、访问网站类别和流量使用情况。通过红黑树的自平衡特性,确保了高效的查找、插入和删除操作。文中提供了C++代码示例,展示了如何实现该算法,并强调其在网络管理中的应用价值。
|
2月前
|
安全 编译器 C++
C++ `noexcept` 关键字的深入解析
`noexcept` 关键字在 C++ 中用于指示函数不会抛出异常,有助于编译器优化和提高程序的可靠性。它可以减少代码大小、提高执行效率,并增强程序的稳定性和可预测性。`noexcept` 还可以影响函数重载和模板特化的决策。使用时需谨慎,确保函数确实不会抛出异常,否则可能导致程序崩溃。通过合理使用 `noexcept`,开发者可以编写出更高效、更可靠的 C++ 代码。
46 1
|
2月前
|
存储 程序员 C++
深入解析C++中的函数指针与`typedef`的妙用
本文深入解析了C++中的函数指针及其与`typedef`的结合使用。通过图示和代码示例,详细介绍了函数指针的基本概念、声明和使用方法,并展示了如何利用`typedef`简化复杂的函数指针声明,提升代码的可读性和可维护性。
95 1
|
3月前
|
自然语言处理 编译器 Linux
|
3月前
|
设计模式 安全 数据库连接
【C++11】包装器:深入解析与实现技巧
本文深入探讨了C++中包装器的定义、实现方式及其应用。包装器通过封装底层细节,提供更简洁、易用的接口,常用于资源管理、接口封装和类型安全。文章详细介绍了使用RAII、智能指针、模板等技术实现包装器的方法,并通过多个案例分析展示了其在实际开发中的应用。最后,讨论了性能优化策略,帮助开发者编写高效、可靠的C++代码。
54 2
|
3月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
220 4
|
3月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
241 13
|
4月前
|
C语言
无头链表二级指针方式实现(C语言描述)
本文介绍了如何在C语言中使用二级指针实现无头链表,并提供了创建节点、插入、删除、查找、销毁链表等操作的函数实现,以及一个示例程序来演示这些操作。
52 0
|
5月前
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
190 4

热门文章

最新文章

推荐镜像

更多