c语言分层理解(c语言指针笔试题解析)(1)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 前言经过了一天的写作,终于完成了这一篇,对指针进行了练习,感觉很爽,分享一下!还望大佬多多关注!1. 一维数组和指针

前言

经过了一天的写作,终于完成了这一篇,对指针进行了练习,感觉很爽,分享一下!还望大佬多多关注!

1. 一维数组和指针

1.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;
}

1.2 解析

#include <stdio.h>
int main()
{
  int a[] = { 1,2,3,4 };
  printf("%d\n", sizeof(a));
  //结果:16字节
  //解释:数组名单独放在sizeof内部计算的是整个数组的大小,单位是字节
  printf("%d\n", sizeof(a + 0));
  //结果:4/8字节
  //解释:数组名并没有单独放在sizeof内部,所以a就是首元素地址,a+0也是首元素地址,所以计算的是一个指针的大小,单位是字节,
  //     千万不能把sizeof(a+0)当作sizeof(a)来计算
  printf("%d\n", sizeof(*a));
  //结果:4字节
  //解释:数组名并没有单独放在sizeof内部,所以a就是首元素地址,对int*指针进行解引用,拿到的是一个int类型的元素,sizeof(*a)=sizeof(int),单位是字节
  printf("%d\n", sizeof(a + 1));
  //结果:4/8字节
  //解释:数组名并没有单独放在sizeof内部,所以a就是首元素地址,因为a是int*指针,所以a+1就跳过一个int类型的元素,
  //     a+1也就是指向数组第二个元素的是指针,所以sizeof(a+1)计算的是指针的大小,单位是字节
  printf("%d\n", sizeof(a[1]));
  //结果:4字节
  //解释:a[1]拿到的是小标为1的元素,元素类型是int类型,所以sizeof(a[1])=sizeof(int),单位是字节
  printf("%d\n", sizeof(&a));
  //结果:4/8字节
  //解释:&a就是取到的整个数组的地址,是一个指向整个数组的指针,所以sizeof(&a)计算的就是指针大小,单位是字节
  printf("%d\n", sizeof(*&a));
  //结果:16字节
  //解释:*&a也就是先拿到数组a的整个数组的地址,再对&a解引用,拿到的是整个数组的大小,所以sizeof(*&a)=sizeof(a),单位是字节
  printf("%d\n", sizeof(&a + 1));
  //结果:4/8字节
  //解释:&a+1也就是拿到整个数组的地址,整个数组的指针类型是一个数组指针类型,&a+1也就跳过一个数组的大小,得到的仍然是地址,地址就是指针,
  //     所以sizeof(&a+1)也就是计算指针的大小,单位是字节
  printf("%d\n", sizeof(&a[0]));
  //结果:4/8字节
  //解释:&a[0]也就是拿到下标为0的元素的地址,也就是拿到第一个元素的地址,地址就是指针,所以sizeof(&a[0])计算的就是指针的大小
  printf("%d\n", sizeof(&a[0] + 1));
  //结果:4/8字节
  //解释:&a[0]+1,也就是拿到第一个元素的地址,再跳过这个元素,拿到下一个元素的地址,地址就是指针,所以sizeof(&a[0]+1)计算的就是指针的大小
  return 0;
}

1.3 输出结果

1.3.1 32位平台

4e1a4618d294f772260cd1b5d053c301.png

1.3.1 64位平台

3095b8de3038c0855f6773405e31900a.png

2. 字符数组和指针

2.1 形式1(char arr[] = {‘#’,‘#’,‘#’,‘#’,‘#’})

2.1.1 例题

#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;
}

2.1.2 解析

#include <stdio.h>
#include <string.h>
int main()
{
  char arr[] = { 'a','b','c','d','e','f' };
  printf("%d\n", sizeof(arr));
  //结果:6字节
  //解释:arr是数组名,单独的数组名在sizeof内部,计算的是数组的大小,数组只分配了6个字节的空间,所以数组大小是6,单位是字节
  printf("%d\n", sizeof(arr + 0));
  //结果:4/8字节
  //解释:arr是数组名,但是并没有单独在sizeof内部,所以arr是首元素地址,arr+0拿到的还是首元素的地址,地址就是指针,所以sizeof计算的是指针的大小,单位是字节
  printf("%d\n", sizeof(*arr));
  //结果:1字节
  //解释:arr是数组名,但是并没有单独在sizeof内部,所以arr是首元素地址,指针类型是char*,对指针解引用就拿到第一个元素,
  //     第一个元素的大小是1个字节,sizeof(*arr)=sizeof(arr[0])
  printf("%d\n", sizeof(arr[1]));
  //结果:1字节
  //解释:arr[1]就是下标为1的元素,也就是数组中第二个元素,元素类型是char,所以sizeof(arr[1])=sizeof(char),单位是字节
  printf("%d\n", sizeof(&arr));
  //结果:4/8字节
  //解释:&arr,拿到的是整个数组的地址,地址就是指针,所以sizeof(&arr)计算的是指针的大小
  printf("%d\n", sizeof(&arr + 1));
  //结果:4/8字节
  //解释:&arr,拿到的是整个数组的地址,地址就是指针,指针类型是数组指针类型,指向的是整个数组,&arr+1也就是跳过一个数组指针的大小,
  //     也就是跳一个数组,得到的仍然是一个地址,地址就是指针,所以sizeof(&arr+1)计算的就是指针的大小,单位是字节
  printf("%d\n", sizeof(&arr[0] + 1));
  //结果:4/8字节
  //解释:&arr[0]得到的是一个下标为0的元素的地址,也就是得到的是数组中第一个元素的地址,地址就是指针,指针类型是int*,&arr[0]+1,
  //     跳过的就是一个整形的大小,得到的仍然是一个地址,地址就是指针,单位是字节
  //strlen库函数计算的是字符串中字符'\0'之前的元素个数
  printf("%d\n", strlen(arr));
  //结果:随机值
  //解释:strlen库函数计算的是字符串中字符'\0'之前的元素个数,但是在数组中我们用的是一块连续的空间,
  //     这块空间的后面都是随机值,arr是数组名,是首元素地址,从数组名这个地址向下找到'\0'字符,在此数组内部这6个元素中没有'\0',
  //     就在数组内存空间的后面继续查找'\0'字符,直到找到'\0'结束,所以这是一个随机值,不确定
  printf("%d\n", strlen(arr + 0));
  //结果:随机值
  //解释:arr是数组名,是首元素地址,arr+0仍然是首元素地址,strlen库函数计算的是字符串中字符'\0'之前的元素个数,
  //     所以要从数组名这个地址向下找到'\0'字符,内存是随机分配的数据,所以不知道'\0'字符的位置,所以是随机值。
  printf("%d\n", strlen(*arr));
  //结果:err
  //解释:arr是首元素地址,*arr就是对第一个元素解引用得到字符'a','a'的ASCII值是97,strlen库函数就是把97当作地址访问,这里是非法访问。
  printf("%d\n", strlen(arr[1]));
  //结果:err
  //解释:arr[1]得到的就是数组的第二个元素'b','b'的ASCII值是98,strlen库函数把98当作地址,非法访问。
  printf("%d\n", strlen(&arr));
  //结果:随机值
  //解释:&arr得到的是整个数组的地址,也就是数组的起始地址,一样也是随机值,不知道'\0'在内存的哪里。
  printf("%d\n", strlen(&arr + 1));
  //结果:随机值
  //解释:&arr得到的是整个数组的地址,地址就是指针,指针类型是一个数组指针类型,&arr+1跳过的是整个数组,同样从数组的末尾地址向下寻找'\0'字符,
  //     任然是个随机值,不确定'\0'在内存中的位置。
  printf("%d\n", strlen(&arr[0] + 1));
  //结果:随机值
  //解释:&arr[0]拿到的是数组第一个元素的地址,地址就是指针,类型是char*,跳过一个char大小,也就得到第二个元素的地址,
  //     在内存中往下寻找'\0'字符,不知道内存中'\0'的位置,所以是随机值。
  return 0;
}

2.1.3 输出结果

说明:因为有两个是error的,所以屏蔽了这两段代码。具体显示是strlen库函数中函数参数是const char*的和这里的char的参数不一致,同时这里的隐藏错误就是非法访问地址。

2.1.3.1 32位平台

17c11fef9e5428d46c80a8dff2f02278.png

2.1.3.2 64位平台

image.png

2.2 形式2(char arr[] = “####”)

2.2.1 例题

#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;
}

2.2.2 解析

#include <stdio.h>
#include <string.h>
int main()
{
  char arr[] = "abcdef";
  printf("%d\n", sizeof(arr));
  //结果:7字节
  //解释:arr数组名单独放在sizeof内部,计算的是整个数组大小,数组包含abcdef\07个字符,所以是7个字节大小
  printf("%d\n", sizeof(arr + 0));
  //结果:4/8字节
  //解释:arr不是单独放在sizeof内部,所以arr是首元素地址,arr+0仍然是首元素地址,地址就是指针,sizeof(arr+0)计算的就是指针的大小,单位是字节
  printf("%d\n", sizeof(*arr));
  //结果:1字节
  //解释:*arr拿到的是第一个元素,元素类型是char,所以sizeof(*arr)=sizeof(char),单位是字节
  printf("%d\n", sizeof(arr[1]));
  //结果:1字节
  //解释:arr[1]拿到的是数组中第二个元素,类型是char,所以sizeof(arr[1])=sizeof(char),单位是字节
  printf("%d\n", sizeof(&arr));
  //结果:4/8字节
  //解释:&arr拿到的是整个数组的地址,地址就是指针,所以sizeof(&arr)计算的就是指针的大小
  printf("%d\n", sizeof(&arr + 1));
  //结果:4/8字节
  //解释:&arr拿到的是整个数组的地址,地址就是指针,指针类型是数组指针,&arr+1跳过的是整个数组,仍然得到的是地址,地址就是指针,所以sizeof(&arr+1)计算的就是指针的大小
  printf("%d\n", sizeof(&arr[0] + 1));
  //结果:4/8字节
  //解释:&arr[0]得到的是一个下标为0的元素的地址,也就是得到的是数组中第一个元素的地址,地址就是指针,指针类型是char*,&arr[0]+1,
  //     跳过的就是一个字符型的大小,得到的仍然是一个地址,地址就是指针,单位是字节
  printf("%d\n", strlen(arr));
  //结果:6字节
  //解释:strlen是计算'\0'之前的元素个数,字符串结束标志就是'\0',元素个数是6,所以strlen(arr)=6,单位是字节
  printf("%d\n", strlen(arr + 0));
  //结果:6字节
  //解释:arr是数组名,也就是首元素地址,从字符串第一个元素向后找到'\0'结束,字符串结束标志是'\0',所以strlen(arr+0)=6,单位是字节
  printf("%d\n", strlen(*arr));
  //结果:err
  //解释:arr是数组名,也就是首元素地址,*arr拿到第一个元素'a','a'的ASCII值是97,strlen库函数就是把97当作地址访问,这里是非法访问。
  printf("%d\n", strlen(arr[1]));
  //结果:err
  //解释:arr[1]得到的就是数组的第二个元素'b','b'的ASCII值是98,strlen库函数把98当作地址,非法访问。
  printf("%d\n", strlen(&arr));
  //结果:6字节
  //解释:&arr得到的是整个数组的地址,也就是数组的起始地址,一样向后找到'\0'停止,strlen(&arr)=6,单位是字节
  printf("%d\n", strlen(&arr + 1));
  //结果:随机值
  //解释:&arr得到的是整个数组的地址,地址就是指针,指针类型是一个数组指针类型,&arr+1跳过的是整个数组,同样从数组的末尾地址向下寻找'\0'字符,
  //     任然是个随机值,不确定'\0'在内存中的位置。
  printf("%d\n", strlen(&arr[0] + 1));
  //结果:5字节
  //解释:&arr[0]拿到的是数组第一个元素的地址,地址就是指针,类型是char*,跳过一个char大小,也就得到第二个元素的地址,
  //     在内存中往下寻找'\0'字符,字符串结束表示'\0',所以strlen(&arr[0] + 1))=5,单位是字节
  return 0;
}

2.2.3 输出结果

说明:说明:因为有两个是error的,所以屏蔽了这两段代码。具体显示是strlen库函数中函数参数是const char*的和这里的char的参数不一致,同时这里的隐藏错误就是非法访问地址。

2.2.3.1 32位平台

947221722186c55056857bc3e9d422eb.png

2.2.3.2 64位平台

f7f769a959424f29877f362897de90b6.png

2.3 形式3(char* p = “####”)

2.3.1 例题

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

2.3.2 解析

#include <stdio.h>
#include <string.h>
int main()
{
  char* p = "abcdef";
  printf("%d\n", sizeof(p));
  //结果:4/8字节
  //解释:p是字符串首元素地址,地址就是指针,指针类型是char*,所以sizeof(p)计算的是指针的大小
  printf("%d\n", sizeof(p + 1));
  //结果:4/8字节
  //解释:p是字符串首元素地址,地址就是指针,指针类型是char*,p+1就跳过一个char大小,也就是得到第二个元素'b'的地址,
  //     地址就是指针,所以sizeof(p+1)计算的就是指针的大小,单位是字节
  printf("%d\n", sizeof(*p));
  //结果:1字节
  //解释:p是字符串首元素地址,地址就是指针,对p解引用拿到'a','a'的类型就是char,所以sizeof(*p)=sizeof(char),单位是字节
  printf("%d\n", sizeof(p[0]));
  //结果:1字节
  //解释:p是字符串首元素地址,也就相当于数组名,p[0]拿到的是第一个元素'a','a'的类型是char,所以sizeof(p[0])=sizeof(char),单位是字节
  printf("%d\n", sizeof(&p));
  //结果:4/8字节
  //解释:p是字符串首元素地址,地址就是指针,&p得到的是一个二级指针,所以sizeof(&p)计算的就是指针的大小,单位是字节
  printf("%d\n", sizeof(&p + 1));
  //结果:4/8字节
  //解释:p是字符串首元素地址,地址就是指针,&p得到的是一个二级指针,二级指针类型是char**,&p+1跳过一个char*大小,得到的仍然是一个地址,地址就是指针,所以siezof(&p+1)计算的就是指针大小
  printf("%d\n", sizeof(&p[0] + 1));
  //结果:4/8字节
  //解释:p是字符串首元素地址,也就相当于数组名,p[0]拿到的是第一个元素'a',&p[0]就是拿到第一个元素的地址,地址就是指针,指针类型是char*,&p[0]+1跳过一个char类型大小,也就是拿到第二个元素'b'的地址,地址就是指针,所以sizeof(&p[0]+1)计算的就是指针大小
  printf("%d\n", strlen(p));
  //结果:6字节
  //解释:p是字符串首元素地址,也就相当于数组名,strlen是计算'\0'之前的元素个数,字符串结束标志就是'\0',元素个数是6,所以strlen(arr)=6,单位是字节
  printf("%d\n", strlen(p + 1));
  //结果:5字节
  //解释:p是首元素地址,地址就是指针,指针类型是char*,p+1跳过一个char大小,得到的是第二个元素'b'的地址,strlen是计算'\0'之前的元素个数,字符串结束标志就是'\0',所以strlen(p+1)=5
  printf("%d\n", strlen(*p));
  //结果:err
  //解释:p是首元素地址,地址就是指针,指针类型是char*,对其解引用拿到'a'字符,'a'的ASCII值是97,strlen库函数就是把97当作地址访问,这里是非法访问。
  printf("%d\n", strlen(p[0]));
  //结果:err
  //解释:p是字符串首元素地址,也就相当于数组名,p[0]拿到的是第一个元素'a','a'的ASCII值是97,strlen库函数就是把97当作地址访问,这里是非法访问。
  printf("%d\n", strlen(&p));
  //结果:随机值
  //解释:p是字符串首元素地址,地址就是指针,&p得到的是一个二级指针,二级指针类型是char**,地址不是从p开始向下寻找了,所以不知道内存中'\0'的位置,所以是随机值,单位是字节
  printf("%d\n", strlen(&p + 1));
  //结果:随机值
  //解释:p是字符串首元素地址,地址就是指针,&p得到的是一个二级指针,二级指针类型是char**,跳过一个char*大小,得到的地址也不是从p开始向下寻找了,
  //     所以不知道内存中'\0'的位置,所以是随机值,单位是字节
  printf("%d\n", strlen(&p[0] + 1));
  //结果:5字节
  //解释:p是字符串首元素地址,也就相当于数组名,p[0]拿到的是第一个元素'a',&p[0]拿到第一个元素'a'的地址,地址就是指针,指针类型是char*,&p[0]+1就是跳过一个char大小,得到第二个元素'b'的地址,从这里往后找,找到'\0'停止,字符串结束标志是'\0',所以strlen(&p[0] + 1))=5,单位是字节
  return 0;
}

2.3.3 输出结果

说明:因为有两个是error的,所以屏蔽了这两段代码。具体显示是strlen库函数中函数参数是const char*的和这里的char的参数不一致,同时这里的隐藏错误就是非法访问地址。

2.3.3.1 32位平台

9bb4339ffba4d1bd3c5eb8b84004bd2e.png

2.3.3.2 64位平台

aeb720b738e5d557bdd6319b1c8f2510.png
































相关文章
|
1月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
167 14
|
1月前
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
665 6
|
1月前
|
存储 网络协议 算法
【C语言】进制转换无难事:二进制、十进制、八进制与十六进制的全解析与实例
进制转换是计算机编程中常见的操作。在C语言中,了解如何在不同进制之间转换数据对于处理和显示数据非常重要。本文将详细介绍如何在二进制、十进制、八进制和十六进制之间进行转换。
60 5
|
1月前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
52 5
|
1月前
|
存储 程序员 C++
深入解析C++中的函数指针与`typedef`的妙用
本文深入解析了C++中的函数指针及其与`typedef`的结合使用。通过图示和代码示例,详细介绍了函数指针的基本概念、声明和使用方法,并展示了如何利用`typedef`简化复杂的函数指针声明,提升代码的可读性和可维护性。
83 0
|
8月前
|
C语言
C语言---指针进阶
C语言---指针进阶
58 0
|
C语言
C语言指针进阶(下)
C语言指针进阶(下)
70 1
|
编译器 C语言
C语言指针进阶(上)
C语言指针进阶(上)
106 1
|
C语言
C语言指针进阶(中)
C语言指针进阶(中)
53 0
|
8月前
|
存储 C语言 C++
C语言指针进阶-1
C语言指针进阶-1
53 1

热门文章

最新文章

推荐镜像

更多