欢迎大家来到贝蒂大讲堂

养成好习惯,先赞后看哦~

所属专栏:C语言学习

贝蒂的主页:Betty‘s blog


1. strlen()和sizeof的区别

名称 区别
sizeof 1. sizeof是操作符
2. sizeof计算操作数所占内存的⼤⼩,单位是字节
3. 不关注内存中存放什么数据
strlen 1. strlen是库函数,使⽤需要包含头⽂件 string.h
2. srtlen是求字符串⻓度的,统计的是 '\0' 之前字符的隔个数
3. 关注内存中是否有'\0' ,如果没有'\0',就会持续往后找,可能会越界

2. 数组名的理解

  1. sizeof(数组名),数组名单独放在括号里,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩。
  2. &数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址。
  3. 除此之外所有的数组名都表⽰⾸元素的地址。

3. 一维数组

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

3.2 输出结果

  • 该环境为VS2022,×64环境。后续也会在该环境下实验

3.3 解析

  1. sizeof(a),这⾥的a是数组名表⽰整个数组,计算整个数组的大小,数组每个元素是整型,一共有四个元素。所以4*4=16
  2. a+0是个表达式,这时a就是数组首元素地址,a+0仍是首元素地址,地址在×86环境下大小为4,在×64环境下大小为8
  3. a是数组首元素地址,对a解引用得到1,1是整型,所以大小为4
  4. a是数组首元素地址,a+1是第二个元素的地址,在×86环境下大小为4,在×64环境下大小为8
  5. a[1]是数组第二个元素2,是一个整型,大小为4
  6. &a是对数组名取地址,代表整个数组的地址,在×86环境下大小为4,在×64环境下大小为8
  7. &a的&和*相互抵消,相当于sizof(a),也就是第一步,大小为16
  8. &a+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,仍是地址,在×86环境下大小为4,在×64环境下大小为8
  9. &a[0]就是数组首元素的地址,在×86环境下大小为4,在×64环境下大小为8
  10. &a[0]+1就是第二个元素地址,与a+1等价,在×86环境下大小为4,在×64环境下大小为8

4. 字符数组

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

(1) 输出结果

(2) 解析

  1. sizeof(arr),这⾥的arr是数组名表⽰整个数组,计算整个数组的大小,数组每个元素是字符型,一共6个元素,1*6=6
  2. a+0是个表达式,这时a就是数组首元素地址,a+0仍是首元素地址,地址在×86环境下大小为4,在×64环境下大小为8
  3. arr是数组首元素地址,对arr解引用得到a,a是字符型,所以大小为1
  4. arr[1]是数组第二个元素b,大小为1
  5. &arr是对数组名取地址,代表整个数组的地址,在×86环境下大小为4,在×64环境下大小为8
  6. &arr+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,仍是地址,在×86环境下大小为4,在×64环境下大小为8
  7. &arr[0]+1就是第二个元素地址,与arr+1等价,在×86环境下大小为4,在×64环境下大小为8

4.2 题目二

int main()
{
//输出结果?
char arr[] = { 'a','b','c','d','e','f' };
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[0] + 1));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
return 0;
}

(1) 输出结果

(2) 解析

  1. strlen是以'\0'为标志的,如果数组里没有,会继续往内存中寻找,直到找到'\0',所以是个随机值
  2. arr+0也是首元素地址,与1同理,所以也是随机值
  3. &arr是对数组名取地址,代表整个数组的地址,为了存储方便也会以数组首元素的地址表示,所以和1.2值相同,也是个随机值
  4. &arr+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,不知道'\0'在哪,所以也是随机值
  5. &arr[0]+1就是第二个元素地址,一直找到内存中的\0',是个随机值且会比1的长度少一
  6. strlen的参数要传地址进去,否则就会出错
  7. strlen的参数要传地址进去,否则就会出错

4.3 题目三

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

(1) 输出结果

(2) 解析

  1. sizeof(arr),计算整个数组的大小,字符串默认的结束标志为'\0',所以数组一共有7个元素,每个元素都是字符型,1*7=7
  2. arr+0是个表达式,这时arr数组首元素地址,arr+0仍是首元素地址,地址在×86环境下大小为4,在×64环境下大小为8
  3. arr是数组首元素的地址,对其解引用等到第一个元素a,a为字符型,大小为1
  4. arr[1]是数组第二个元素b,大小也为1
  5. &arr是对数组名取地址,代表整个数组的地址,在×86环境下大小为4,在×64环境下大小为8
  6. &arr+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,仍是地址,在×86环境下大小为4,在×64环境下大小为8
  7. &arr[0]+1就是第二个元素地址,与arr+1等价,在×86环境下大小为4,在×64环境下大小为8

4.4 题目四

int main()
{
//输出结果?
char arr[] = "abcdef";
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[0] + 1));
printf("%d\n", strlen(*arr));
printf("%d\n", strlen(arr[1]));
return 0;
}

(1) 输出结果

(2) 解析

  1. strlen是以'\0'为标志的,字符串默认结束标志位'\0',所以是6
  2. arr+0也是首元素地址,与1同理,所以也是6
  3. &arr是对数组名取地址,代表整个数组的地址,为了存储方便也会以数组首元素的地址表示,所以和1.2值相同,也是6
  4. &arr+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,不知道'\0'在哪,所以是随机值
  5. &arr[0]+1就是第二个元素地址,一直找到内存中的\0',会比1的长度少一,是5
  6. strlen的参数要传地址进去,否则就会出错
  7. strlen的参数要传地址进去,否则就会出错

4.5 题目五

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 + 1));
printf("%d\n", sizeof(&p[0] + 1));
printf("%d\n", sizeof(*p));
printf("%d\n", sizeof(p[0]));
return 0;
}

(1) 输出结果

(2) 解析

  1. 字符串存储的是首元素地址,所以p相当于首元素地址,在×86环境下大小为4,在×64环境下大小为8
  2. 同理p+1是第二个元素的地址,在×86环境下大小为4,在×64环境下大小为8
  3. 对p取地址,相当于得到还是地址,在×86环境下大小为4,在×64环境下大小为8
  4. &arr+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,仍是地址,在×86环境下大小为4,在×64环境下大小为8
  5. &p[0]+1就是第二个元素地址,在×86环境下大小为4,在×64环境下大小为8
  6. *p与p[0]都是第一个元素,大小为1

4.6 题目六

int main()
{
//输出结果?
char* p = "abcdef";
printf("%d\n", strlen(p));
printf("%d\n", strlen(p + 1));
printf("%d\n", strlen(&p));
printf("%d\n", strlen(&p + 1));
printf("%d\n", strlen(&p[0] + 1));
printf("%d\n", strlen(*p));
printf("%d\n", strlen(p[0])); return 0;
}

(1) 输出结果

(2) 解析

  1. strlen是以'\0'为标志的,字符串默认结束标志位'\0',所以是6
  2. p+1就是第二个元素地址,一直找到内存中的\0',会比1的长度少一,是5
  3. &p是对数组名取地址,代表整个数组的地址,为了存储方便也会以数组首元素的地址表示,所以和1值相同,也是6
  4. &p+1指的是以整个数组的地址为单位,跳过整个数组之后的地址,不知道'\0'在哪,所以是随机值
  5. &p[0]+1就是第二个元素地址,一直找到内存中的\0',会比1的长度少一,是5
  6. *p与p[0]都是第一元素。strlen的参数要传地址进去,否则就会出错

5. 二维数组

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

(1) 输出结果

(2) 解析

  1. sizeof(a),这⾥的a是数组名表⽰整个数组,计算整个二维数组的大小,数组每个元素是整型型,一共12个元素,12*4=48
  2. a[0][0]是指二维数组第一个元素,是个整型,大小为4
  3. a[0]相当于第一排首元素的地址,在二维数组中这里可以抽象理解为一维数组的数组名,单独放在sizeof中,所以4*4=16
  4. a[0]+1是一个表达式,代表第一排第二个元素的地址,在×86环境下大小为4,在×64环境下大小为8
  5. a[0]相当于第一排首元素的地址,*a[0]就是首元素,大小为4
  6. a第一排的地址,a+1是第二排地址,在×86环境下大小为4,在×64环境下大小为8
  7. *(a+1)等价于a[1],相当于第二排的数组名,4*4=16
  8. &a[0] + 1等价于a=1,即第二排地址,在×86环境下大小为4,在×64环境下大小为8
  9. *(&a[0] + 1))等价于*(a+1),与7相同,大小为16
  10. *a相当于第一排的数组名,大小也为16
  11. 因为sizeof只是根据类型判断,所以不会管是否越界,所以仍相当于一排的数组名,大小为16

6. 指针深度理解

6.1 题目一

#include <stdio.h>
int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int* ptr = (int*)(&a + 1);
printf("%d,%d", *(a + 1), *(ptr - 1));
return 0;
}

(1) 输出结果

(2) 解析

  1. a是首元素的地址,a+1是指第二个元素的地址,对其解引用就是第二个元素,也就是2
  2. &a+1跳过整个数组,在被强制类型转换为int*,减1指向5,解引用就是5

6.2 题目二

#include<stdio.h>
//在X86环境下
//假设结构体的⼤⼩是20个字节
//程序输出的结构是啥?
struct Test
{
int Num;
char* pcName;
short sDate;
char cha[2];
short sBa[4];
}*p = (struct Test*)0x100000;
int main()
{
printf("%p\n", p + 0x1);
printf("%p\n", (unsigned long)p + 0x1);
printf("%p\n", (unsigned int*)p + 0x1);
return 0;
}

(1) 输出结果

(2) 解析

  1. p的地址是0x100000,是十六进制表示,加1跳过一个结构体大小(20),十六进制表示就是00100014
  2. p被强制类型转换为无符号长整型,加1相当于加上数字1,就为00100001
  3. p被强制类型转换为无符号整型的指针,加1跳过一个整型,为00100004

6.3 题目三

#include <stdio.h>
//输出什么?
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int* p;
p = a[0];
printf("%d", p[0]);
return 0;
}

(1) 输出结果

(2) 解析

  1. 括号表达式从左往右依次计算,取最后一次的值,所以数组中元素简化为1,3,5
  2. p[0]就是数组的首元素,为1

6.4 题目四

//假设环境是x86环境,程序输出的结果是啥?
#include <stdio.h>
int main()
{
int a[5][5];
int(*p)[4];
p = a;
printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
return 0;
}

(1) 输出结果

(2) 解析

  1. 首先我们得知道数组在内存中是连续存储的
  2. p的类型为int(*p)[4],p[4][2]等价于*(*(p+4)+2),也就是说把p跳过4个以int(*p)[4]类型的距离,跳过2个整型
  3. 示意图如下,蓝色代表p[4][2],红色代表a[4][2]

  1. &p[4][2] - &a[4][2]之间差四个元素,值为-4,又因为地址是无符号的整数,所以发生整型提升

6.5 题目五

#include <stdio.h>
//输出什么?
int main()
{
char* a[] = { "work","at","alibaba" };
char** pa = a;
pa++;
printf("%s\n", *pa);
return 0;
}

(1) 输出结果

(2) 解析

  1. 这是一个指针数组,每个元素类型为char*,分别指向一个字符串,如图

  1. pa++跳过一个char*的地址指向第二个元素,对其解引用得到at

6.6 题目六

#include <stdio.h>
int main()
{
char* c[] = { "ENTER","NEW","POINT","FIRST" };
char** cp[] = { c + 3,c + 2,c + 1,c };
char*** cpp = cp;
printf("%s\n", **++cpp);
printf("%s\n", *-- * ++cpp + 3);
printf("%s\n", *cpp[-2] + 3);
printf("%s\n", cpp[-1][-1] + 1);
return 0;
}

(1) 输出结果

(2) 解析

  1. 由图可知cpp++指向c+2,再两次解引用得到"POINT"的首元素地址

  1. cpp先++指向c+1,解引用得到c+1,再--得到c解引用得到ENTER的首元素地址,+3得到E的地址

  1. *cpp[-2]等价于**(cpp-2),cpp-2指向c+3,两次解引用得到FIRST的首元素地址,+3得到S的地址

  1. cpp[1][-1]等价于*(*(cpp-1)-1),这时候cpp指向的是第一个c,cpp-1再解引用得到c+2,再-1解引用得到NEW的地址,最好加1得到E的地址

揭秘C语言的心脏:深入探索指针与数组的奥秘的更多相关文章

  1. C语言程序设计(十一) 指针和数组

    第十一章 指针和数组 一旦给出数组的定义,编译系统就会为其在内存中分配固定的存储单元,相应的,数组的首地址也就确定了 C语言中的数组名有特殊的含义,它代表存放数组元素的连续存储空间的首地址 //L11 ...

  2. C语言 第八章 函数、指针与宏

    一.函数 函数是一个包含完成一定功能的执行代码段.我们可以把函数看成一个"黑盒子", 你只要将数据送进去就能得到结果, 而函数内部究竟是如何工作的的, 外部程序是不知道的.外部程序 ...

  3. 指针与数组的区别 —— 《C语言深度剖析》读书心得

    原书很多已经写的很清楚很精炼了,我也无谓做无意义的搬运,仅把一些基础和一些我自己以前容易搞混的地方写一下. 1. 意义: 指针: 指针也是一种类型,长度为4字节,其存放的内容只能是一个地址(4字节). ...

  4. 深入理解C语言中的指针与数组之指针篇

    转载于http://blog.csdn.net/hinyunsin/article/details/6662851     前言 其实很早就想要写一篇关于指针和数组的文章,毕竟可以认为这是C语言的根本 ...

  5. C语言指针与数组的定义与声明易错分析

    部分摘自<C语言深度解剖> 1.定义为数组,声明为指针 在文件1中定义: char a[100]; 在文件2中声明: extern char *a; //这样是错误的 这里的extern告 ...

  6. 深入理解C语言中的指针与数组之指针篇(转载)

    前言 其实很早就想要写一篇关于指针和数组的文章,毕竟可以认为这是C语言的根本所在.相信,任意一家公司如果想要考察一个人对C语言的理解,指针和数组绝对是必考的一部分. 但是之前一方面之前一直在忙各种事情 ...

  7. C语言之指针与数组总结

    和指针相关的问题口诀1: 1. 地址变量得地址,得谁地址指向谁 和指针相关的问题要画图: 内容变量画房子,指针画箭头 ---->口 ------------------------------- ...

  8. C语言的本质(11)——指针与数组

    1.指针数组和数组指针的内存布局 初学者总是分不出指针数组与数组指针的区别.其实很好理解:指针数组:首先它是一个数组,数组的元素都是指针,数组占多少个字节由数组本身决定.它是"储存指针的数组 ...

  9. 娓娓道来c指针 (0)c语言的梦魇:c指针

    (0)c语言的梦魇:c指针 序 c语言中有一个重点:c指针.它也是一个难点.当然,这是一句废话:重点往往也是难点.在c标准中,对指针的定义是这种: 指针的类型是derived from其他类型,也就是 ...

  10. C语言指针和数组知识总结(上)

    C语言指针和数组知识总结(上) 一.指针的基础 1.C语言中,变量的值能够通过指针来改变,打印指针的语句符号可以是:  %08x 2.指针的本质 指针的本质就是变量,那么既然是变量,那么一定会分配地址 ...

随机推荐

  1. Educational Codeforces Round 106 (Rated for Div. 2) 简单题解(A~C)

    1499A. Domino on Windowsill 题意:给定一个 \(2 \times n\) 的空间,\(k1.k2 行要设置为白色(2 \times 1)\) 然后其他的设置为黑色 思路:为 ...

  2. postman+springboot一次上传多个文件

    开发中到前端一次上传多个文件的需求如何实现,下面使用postman模拟前端的请求,后端使用srpingboot来实现 1.postman设置 2.Java代码 @RestController @Req ...

  3. 六、docker数据卷

    系列导航 一.docker入门(概念) 二.docker的安装和镜像管理 三.docker容器的常用命令 四.容器的网络访问 五.容器端口转发 六.docker数据卷 七.手动制作docker镜像 八 ...

  4. <vue初体验> 基础知识 3、vue的计数器

    系列导航 <vue初体验> 一. vue的引入和使用体验 <vue初体验> 二. vue的列表展示 <vue初体验> 三. vue的计数器 <vue初体验&g ...

  5. 我发现明显产品bug啦

    1.  百度云在下载时,如果选中的文件过多,在点击下载后,不能即时取消所有的下载项! 如下图,点击""全部取消" 出现在列表中项全部消失,但后续新的项继续出现,仍在下载, ...

  6. python常见面试题讲解(一)字符串最后一个单词的长度

    题目描述 计算字符串最后一个单词的长度,单词以空格隔开. 输入描述: 一行字符串,非空,长度小于5000. 输出描述: 整数N,最后一个单词的长度. 示例1 输入 复制 hello world 输出 ...

  7. python常见面试题讲解(四)字符串分隔

    题目描述 •连续输入字符串,请按长度为8拆分每个字符串后输出到新的字符串数组:•长度不是8整数倍的字符串请在后面补数字0,空字符串不处理. 输入描述: 连续输入字符串(输入2次,每个字符串长度小于10 ...

  8. 百度网盘(百度云)SVIP超级会员共享账号每日更新(2023.11.21)

    百度网盘会员账号共享(11.21更新) 账号:tgc91660 密码:6858hykh 账号:ofj51327 密码:rvzp3251 账号:5799osrb 密码:862lwtr 账号:3730sw ...

  9. [转帖]被误解的CPU利用率、超线程、动态调频 —— CPU 性能之迷 Part 1

    https://blog.mygraphql.com/zh/notes/hw/hyper-threading/ 引 性能测试.压力测试.业务系统性能容量评估.这 3 件事,可以认为是大部分程序员/软件 ...

  10. [转帖]可直接拿来用的kafka+prometheus+grafana监控告警配置

    kafka配置jmx_exporter 点击:https://github.com/prometheus/jmx_exporter,选择下面的jar包下载: 将下载好的这个agent jar包上传到k ...