深入理解指针(6)

1. sizeof和strlen的对比

1.1sizeof

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

#include <stdio.h>
int main()
{
	int a = 10;
	printf("%zd\n", sizeof(a));
	printf("%zd\n", sizeof a);
	printf("%zd\n", sizeof(int));
	return 0;
}

运行结果:
在这里插入图片描述

1.2 strlen

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

size_t strlen ( const char * str );

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

#include <stdio.h>
int main()
{
	char arr1[3] = { 'a', 'b', 'c' };//a b c
	char arr2[] = "abc";//a b c \0
	printf("%d\n", strlen(arr1));//随机值,没有\0
	printf("%d\n", strlen(arr2));

	printf("%zd\n", sizeof(arr1));
	printf("%zd\n", sizeof(arr2));//有\0
	return 0;
}

运行结果:
在这里插入图片描述

1.3 sizeof 和 strlen的对⽐

在这里插入图片描述

2. 数组和指针笔试题解析

2.1 ⼀维数组

//* sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表示整个数组,计算的是整个数组的大小,单位是字节
//*& 数组名,这⾥的数组名表示整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素的地址是有区别的)
// 除此之外,任何地⽅使⽤数组名,数组名都表示⾸元素的地址。

//地址 x86 环境下4个字节  x64 环境下8个字节
int main()
{
	int a[] = { 1,2,3,4 };
	printf("%zd\n", sizeof(a));//这里的a表示数组的大小 16
	printf("%zd\n", sizeof(a + 0));//数组名a是首元素地址,a+0还是首元素的地址,这里表示地址 4/8
	printf("%zd\n", sizeof(*a));//数组名a是首元素地址,*a是首元素,第一个元素大小 4
	printf("%zd\n", sizeof(a + 1));//数组名a是首元素地址,a+1是第二个元素的地址 4/8
	printf("%zd\n", sizeof(a[1]));//第二个元素大小 4
	printf("%zd\n", sizeof(&a));//&a是数组的地址,数组的地址也是地址,表示地址 4/8
	printf("%zd\n", sizeof(*&a));//就是a,表示数组的大小 16
 //sizeof(*&a) --> sizeof(a) - 16
 //&a --> int (*) [4]
	printf("%zd\n", sizeof(&a + 1));//&a+1相对于&a是跳过整个数组,但是即使跳过整个数组,&a+1依然是地址,表示地址 4/8
	printf("%zd\n", sizeof(&a[0]));//&a[0]首元素的地址 4/8
	printf("%zd\n", sizeof(&a[0] + 1));//&a[0]首元素的地址,&a[0] + 1是第二个元素的地址 4/8
	return 0;
}

运行结果:
在这里插入图片描述

2.2 字符数组

代码1:

int main()
{
	char arr[] = { 'a','b','c','d','e','f' };
	printf("%zd\n", sizeof(arr));//数组名单独放在sizeof内部,这里的arr表示整个数组,计算整个数组的大小,单位是字节  6
	printf("%zd\n", sizeof(arr + 0));//arr表示数组首元素的地址,arr + 0还是首元素的地址,是地址就是 4/8 个字节
	printf("%zd\n", sizeof(*arr));//arr表示首元素的地址,*arr就是首元素,大小是 1
	printf("%zd\n", sizeof(arr[1]));//arr[1]就是第二个元素,大小是 1
	printf("%zd\n", sizeof(&arr));//&arr是数组的地址,数组的地址还是地址,是地址就是 4/8
	printf("%zd\n", sizeof(&arr + 1));//&arr + 1是跳过整个数组后的地址,是地址就是 4/8
	printf("%zd\n", sizeof(&arr[0] + 1));//第二个元素的地址, 是 4/8
}

运行结果:
在这里插入图片描述

strlen函数介绍
在这里插入图片描述

代码2:

//string是库函数
//包含头文件#include <string.h>
//求字符串长度的,统计的是在字符串中\0之前的字符的个数
//如果没有\0就会一直往后找
int main()
{
	char arr[] = { 'a','b','c','d','e','f' };
	printf("%zd\n", strlen(arr));//因为字符串数组arr中没有\0,所以在求字符串长度的时候,会一直往后找,产生的结果是 随机值
	printf("%zd\n", strlen(arr + 0));//arr+0是首元素的地址,和第一个一样也是 随机值
	//printf("%zd\n", strlen(*arr));//err(错误),arr是数组首元素的地址,*arr是数组首元素,就是'a'-97
	//strlen函数参数的部分需要传一个地址,当我们传的是字符'a'时,'a'的ASICC值是97,那就是将97作为地址传参
	//strlen就会从97这个地址开始统计字符串长度,这就非法访问了
	//printf("%zd\n", strlen(arr[1]));//err
	printf("%zd\n", strlen(&arr));//&arr是数组的地址,数组的地址和数组首元素的地址,值是一样的,
	//那么传给strlen函数后,依然是从数组的第一个元素位置往后统计
	printf("%zd\n", strlen(&arr + 1));//随机值
	printf("%zd\n", strlen(&arr[0] + 1));//&arr[0] + 1第二个元素的地址,也是 随机值
	return 0;
}

运行结果:
在这里插入图片描述

代码3:

int main()
{
	char arr[] = "abcdef";//a b c d e f \0
	printf("%zd\n", sizeof(arr));//7
	printf("%zd\n", sizeof(arr + 0));//arr+0首元素的地址,地址 4/8
	printf("%zd\n", sizeof(*arr));//*arr就是首元素, 1
	//*arr --> *(arr+0) --> arr[0]
	printf("%zd\n", sizeof(arr[1]));//arr[1]就是第二个元素, 1
	printf("%zd\n", sizeof(&arr));//&arr是数组地址,地址 4/8
	printf("%zd\n", sizeof(&arr + 1));//&arr + 1 是跳过一个数组的地址, 地址 4/8
	printf("%zd\n", sizeof(&arr[0] + 1));//&arr[0] + 1是第二个元素的地址, 地址 4/8
	return 0;
}

运行结果:
在这里插入图片描述

代码4:

int main()
{
	char arr[] = "abcdef";
	printf("%zd\n", strlen(arr));//6
	printf("%zd\n", strlen(arr + 0));//6
	//printf("%zd\n", strlen(*arr));//err
	//printf("%zd\n", strlen(arr[1]));//err
	printf("%zd\n", strlen(&arr));//6
	printf("%zd\n", strlen(&arr + 1));//随机值
	printf("%zd\n", strlen(&arr[0] + 1));//5
	return 0;
}

运行结果:
在这里插入图片描述

代码5:

int main()
{
	char* p = "abcdef";
	printf("%zd\n", sizeof(p));//p是一个指针变量, 地址 4/8
	printf("%zd\n", sizeof(p + 1));//p+1是'b'的地址,地址 4/8
	printf("%zd\n", sizeof(*p));//*p就是'a', 1
	printf("%zd\n", sizeof(p[0]));//p[0] --> *(p+0) --> *p  1
	printf("%zd\n", sizeof(&p));// 4/8
	printf("%zd\n", sizeof(&p + 1));// 4/8
	printf("%zd\n", sizeof(&p[0] + 1));//&p[0] + 1就是'b'的地址, 4/8
	return 0;
}

运行结果:
在这里插入图片描述

代码6:

int main()
{
	char* p = "abcdef";
	printf("%zd\n", strlen(p));//6
	printf("%zd\n", strlen(p + 1));//5
	//printf("%zd\n", strlen(*p));//err
	//printf("%zd\n", strlen(p[0]));//err
	printf("%zd\n", strlen(&p));//随机值
	printf("%zd\n", strlen(&p + 1));//随机值
	printf("%zd\n", strlen(&p[0] + 1));//5
	return 0;
}

运行结果:
在这里插入图片描述

2.3 ⼆维数组

int main()
{
	int a[3][4] = { 0 };
	printf("%zd\n", sizeof(a));//3*4*4  48
	printf("%zd\n", sizeof(a[0][0]));// 4
	printf("%zd\n", sizeof(a[0]));//a[0]是第一行这个一维数组的数组名
	//数组名算是单独放在sizeof内部,计算的是整个数组的大小,16
	printf("%zd\n", sizeof(a[0] + 1));
	//a[0]是第一行的数组名,没有单独放在sizeof内部,没有&
	//a[0]表示数组首元素的地址,也就是a[0][0]的地址
	//所以a[0] + 1是第一行第二个元素的地址,地址 4/8
	printf("%zd\n", sizeof(*(a[0] + 1)));// 4
	//计算的是第一行第二个元素的大小
	printf("%zd\n", sizeof(a + 1));// 4/8
	//a是数组首元素的地址,是第一行的地址 int (*) [4]
	//a+1就是第二行的地址
	printf("%zd\n", sizeof(*(a + 1)));//16
	//*(a + 1) --> arr[1] --> sizeof(*(a + 1)) -> sizeof(a[1]) 计算的是第二行的大小
	//a + 1 --> 是第二行的地址,int (*) [4]
	//*(a + 1) 访问第二行的数组
	printf("%zd\n", sizeof(&a[0] + 1));// 4/8
	//&a[0]是第一行的地址
	//&a[0]+1是第二行的地址
	printf("%zd\n", sizeof(*(&a[0] + 1)));//计算的是第二行的大小 16
	printf("%zd\n", sizeof(*a));//计算的是第一行的大小 16
	//a是数组首元素的地址,就是第一行的地址
	//*a 就是第一行
	//*a --> *(a+0) --> a[0]
	printf("%zd\n", sizeof(a[3]));//16
	//a[3] --> a[4]
	//sizeof并不会真正的计算
	return 0;
}

运行结果:
在这里插入图片描述
数组名的意义:

  1. sizeof(数组名),这⾥的数组名表⽰整个数组,计算的是整个数组的大小。
  2. &数组名,这⾥的数组名表示整个数组,取出的是整个数组的地址。
  3. 除此之外所有的数组名都表示⾸元素的地址。

2.3.1 sizeof会不会真正计算

int main()
{
	int a = 7;
	short s = 4;
	printf("%zd\n", sizeof(s = a + 3));
	printf("%d\n", s);
}

运行结果:
在这里插入图片描述
总结:sizeof并不会真正计算

3. 指针运算笔试题解析

3.1 题⽬1:

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

运行结果:
在这里插入图片描述

分析:
在这里插入图片描述

3.2 题⽬2

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

运行结果:
在这里插入图片描述

3.2 题⽬3

int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
	//逗号表达式
	//实际存放的是  1 3 5 0 0 0
	int* p;
	p = a[0];
	printf("%d", p[0]);
	return 0;
}

运行结果:
在这里插入图片描述

3.2 题⽬4

//假设环境是x86环境,程序输出的结果是啥?
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]);
	//第一个打印的是指针 -4 
	//10000000000000000000000000000100 原码
	//11111111111111111111111111111100 反码
	//FF FF FF FC 16进制输出

	//指针相减表示元素个数
	//但现在是小的减大的所以是 -4
	return 0;
}

运行结果:
在这里插入图片描述
分析:
在这里插入图片描述

3.5 题⽬5

int main()
{
 int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
 int *ptr1 = (int *)(&aa + 1);
 int *ptr2 = (int *)(*(aa + 1));
 printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));
 return 0;
}

运行结果:
在这里插入图片描述

3.6 题⽬6

int main()
{
 char *a[] = {"work","at","alibaba"};
 char**pa = a;
 pa++;
 printf("%s\n", *pa);
 return 0;
}

运行结果:
在这里插入图片描述

3.7 题⽬7

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

运行结果:
在这里插入图片描述
分析:
在这里插入图片描述

  • 46
    点赞
  • 35
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 18
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 18
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蹋雾

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值