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

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 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[整数]都表示第几行的地址。

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

 

目录
相关文章
|
15天前
|
搜索推荐 编译器 C语言
【C++核心】特殊的元素集合-数组与字符串详解
这篇文章详细讲解了C++中数组和字符串的基本概念、操作和应用,包括一维数组、二维数组的定义和使用,以及C风格字符串和C++字符串类的对比。
55 4
|
2天前
|
存储 安全 编译器
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值(一)
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值
|
7天前
|
存储 C++ 索引
C++函数指针详解
【10月更文挑战第3天】本文介绍了C++中的函数指针概念、定义与应用。函数指针是一种指向函数的特殊指针,其类型取决于函数的返回值与参数类型。定义函数指针需指定返回类型和参数列表,如 `int (*funcPtr)(int, int);`。通过赋值函数名给指针,即可调用该函数,支持两种调用格式:`(*funcPtr)(参数)` 和 `funcPtr(参数)`。函数指针还可作为参数传递给其他函数,增强程序灵活性。此外,也可创建函数指针数组,存储多个函数指针。
|
16天前
|
编译器 C++
【C++核心】指针和引用案例详解
这篇文章详细讲解了C++中指针和引用的概念、使用场景和操作技巧,包括指针的定义、指针与数组、指针与函数的关系,以及引用的基本使用、注意事项和作为函数参数和返回值的用法。
17 3
|
2天前
|
存储 编译器 程序员
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值(二)
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值
|
2天前
|
Serverless 编译器 C语言
【C语言】指针篇- 深度解析Sizeof和Strlen:热门面试题探究(5/5)
【C语言】指针篇- 深度解析Sizeof和Strlen:热门面试题探究(5/5)
|
2天前
|
编译器 C语言
【C语言】指针篇-深入探索数组名和指针数组- 必读指南(2/5)
【C语言】指针篇-深入探索数组名和指针数组- 必读指南(2/5)
|
1月前
|
C++
C++(十一)对象数组
本文介绍了C++中对象数组的使用方法及其注意事项。通过示例展示了如何定义和初始化对象数组,并解释了栈对象数组与堆对象数组在初始化时的区别。重点强调了构造器设计时应考虑无参构造器的重要性,以及在需要进一步初始化的情况下采用二段式初始化策略的应用场景。
|
1月前
|
C++
C++(十八)Smart Pointer 智能指针简介
智能指针是C++中用于管理动态分配内存的一种机制,通过自动释放不再使用的内存来防止内存泄漏。`auto_ptr`是早期的一种实现,但已被`shared_ptr`和`weak_ptr`取代。这些智能指针基于RAII(Resource Acquisition Is Initialization)原则,即资源获取即初始化。RAII确保对象在其生命周期结束时自动释放资源。通过重载`*`和`-&gt;`运算符,可以方便地访问和操作智能指针所指向的对象。

推荐镜像

更多