(目录)
一、知识回顾
1、回顾sizeof()、strlen的作用:
sizeof()
是用来求取变量
或者类型
所占内存空间的大小(单位:字节
)。- sizeof计算的是占用内存空间的大小,单位是字节,不关注内存中到底存放的是什么。sizeof不是函数,是操作符
strlen()
是一个库函数是专门用来计算字符串
长度的,在对其进行调用前是需要包含头文件<string.h>。- strlen()函数是通过字符串结束标志
\0
来计算字符串长度的,但计算出来的字符串长度是不包括\0
的,也就是说所谓的长度就是\0 前字符的个数
。strlen是函数
sizeof是一个运算符,用于计算一个变量或数据类型所占的字节数,不论变量中存储的数据是否为空。而strlen是一个函数,用于计算一个字符串中的字符数,不包括字符串末尾的空字符'\0'。 另外,sizeof可以用于计算任意数据类型的大小,包括基本数据类型和自定义数据类型。 而strlen只能用于计算字符串的长度。 因此,
sizeof通常用于确定数组所占的存储空间,而strlen用于确定字符串的长度。
2、数组和指针
- 数组 - 能够存放一组相同类型的元素,数组的大小取决于数组的元素个数和元素类型。
- 指针就是地址,大小都是4/8。(32位机器是4字节,64位机器是8字节)
3、数组名
- 大部分情况下
数组名是首元素地址
。 - 2个例外
sizeof(数组名)
—— 数组名表示整个数组,计算的是整个数组的大小。(看数组名是否是单独放在sizeof()里面)
&数组名
————-数组名表示整个数组,取出的是数组的地址
二、sizeof()、strlen()的对比
1、注意区别:
- 字符数组 - char arr1[]={'a','b','c','d','e','f'} ->[a,b,c,d,e,f]
- 字符串数组 - char arr2[]="abcdef" ->[a,b,c,d,e,f,
/0
]
- sizeof()计算的是占用内存空间的大小,单位是字节,
不关注内存中到底存放的是什么
。 - strlen()是针对字符串的,求的是字符串的长度,本质上统计的是
/0
之前出现的字符个数。
2、一维数组与一级指针
整型数组
int a[] = { 1,2,3,4 };
printf("%d\n", sizeof(a[1]));//4
printf("%d\n", sizeof(a));//16
//sizeof(a) 就是数组名单独放在sizeof内部 计算的是整个数组的大小。
printf("%d\n", sizeof(a + 0));// 4/8
//a + 0 a不是单独放在sizeof()里面,则是数组首元素的地址,sizeof则是计算数组首元素地址的大小
printf("%d\n", sizeof(*a));//4
//a不是单独放在sizeof()里面,则是数组首元素的地址 *a == a[0],sizeof()则是计算首元素的大小。
//*a -> *&a[0] -> a[0]
printf("%d\n", sizeof(a + 1));// 4/8
//a不是单独放在sizeof()里面,则是数组首元素的地址 -- int*
//a+1 跳过1个整型,是第二个元素的地址 sizeof()是计算第二个元素地址的大小。
printf("%d\n", sizeof(&a));/ /4/8
//&a - 取出的是数组的地址,但是数组的地址也是地址,地址的的大小就是4/8。
//int (*pa)[4] = &a;
printf("%d\n", sizeof(*&a));//16
//等于sizeof(a)
printf("%d\n", sizeof(&a+1));// 4/8
//&a --> int (*)[4],取出的是数组的地址。
//&a + 1 则要跳过一个数组。
printf("%d\n", sizeof(&a[0]));
//取出首元素地址,sizeof计算的是元素的地址4/8
printf("%d\n", sizeof(&a[0]+1));//4/8
//取出的是第二个元素的地址,sizeof()计算的是地址的大小。
字符数组
char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));//6
printf("%d\n", sizeof(arr + 0));// 4/8
//arr不单独放在sizeof里面,是数组首元素的地址 ,计算的大小为4/8
printf("%d\n", sizeof(*arr));//1
//arr不单独放在sizeof里面,arr是首元素地址,*arr 是首元素,大小为1
printf("%d\n", sizeof(arr[1]));//元素'b'
printf("%d\n", sizeof(&arr));// 4/8
//&arr 是数组的地址,地址的大小的都是 4/8
printf("%d\n", sizeof(&arr + 1));//4/8
printf("%d\n", sizeof(&arr[0] + 1));//4/8
//*****************************************
printf("%d\n", strlen(arr));//随机值
//arr数组首元素地址,找不到‘\0’,所以是计算的大小是随机值
printf("%d\n", strlen(arr + 0));//随机值,同上
printf("%d\n", strlen(*arr));//非法访问
//strlen传入char*(地址)
//*arr = 'a'——>(97)被当成地址,非法访问
printf("%d\n", strlen(arr[1]));//非法访问,同上
printf("%d\n", strlen(&arr));//随机值
//找不到/0
printf("%d\n", strlen(&arr + 1));//随机值 - 6
printf("%d\n", strlen(&arr[0] + 0));//随机值 - 1
字符串数组
char arr[]="abcdef";//[a b c d e f /0]
printf("%d\n",sizeof(arr));//整个数组大小 7
printf("%d\n",sizeof(arr+0));//首元素地址大小 4/8
printf("%d\n",sizeof(*arr));//首元素大小 1
//*arr = arr[0] = *(arr+0)
printf("%d\n",sizeof(arr[1]));//1
printf("%d\n",sizeof(&arr));//数组的地址,大小 4/8
printf("%d\n",sizeof(&arr+1));//+1 跳过一个数组的地址,大小 4/8
printf("%d\n",sizeof(&arr[0]+1));//+1 跳过一个元素的地址,大小 4/8
//******************************************************************
printf("%d\n",strlen(arr));//arr表示首元素地址,计算得到数组大小为 6
printf("%d\n",strlen(arr+0));//6,同上
printf("%d\n",strlen(*arr));//非法访问
printf("%d\n",strlen(arr[1]));//非法访问
//strlen是传入char*地址,*arr和arr[1]是数组的元素,传给strlen会被当做地址,造成非法访问
printf("%d\n",strlen(&arr));//&arr表示整个数组的大小,6
printf("%d\n",strlen(&arr+1));//数组的地址 +1后就找不到 /0 ,大小为随机值
printf("%d\n",strlen(&arr[0]+1));//从第二个元素开始算,大小为5
指针
char* p="abcdef";
printf("%d\n",sizeof(p));//4/8
//指针变量p 存放着的是地址,所以大小为4/8
printf("%d\n",sizeof(p+1));//字符’b‘的地址,大小为4/8
printf("%d\n",sizeof(*p));
//解引用的一个char类型元素,大小为1
printf("%d\n",sizeof(p[0]));//1
//p[0] = *(p+0)
printf("%d\n",sizeof(&p));//4/8
printf("%d\n",sizeof(&p+1));
//+1跳过一个char*地址,还是地址,大小还是4/8
//********************************************
printf("%d\n",strlen(p));//大小为 6
printf("%d\n",strlen(p+1));//p+1是'b'的地址,从'b'开始算,大小为5
printf("%d\n",strlen(*p));//*p为'a',不是地址,非法访问
printf("%d\n",strlen(&p));//随机值
//&p取的是变量p的地址,不是p里面存放的地址,所以无法找打字符串
printf("%d\n",strlen(&p+1));//随机值,同上
printf("%d\n",strlen(&p[0]+1));//大小为 5
//p[0]是’a‘,&p[0]去'a'的地址,+1的'b'的地址,从'b'开始计算
3、二维数组与二级指针
整型数组
int a[3][4]={0};//三行四列的数组
printf("%d\n",sizeof(a));//4*3*4=48
//a这个数组名单独放在sizeof里面,表示整个数组,计算的是整个数组的大小
printf("%d\n",sizeof(a[0][0]));//第一行第一个元素,大小4字节
printf("%d\n",sizeof(a[0]));//16
//a[0] 是第一行的数组名,这是数组名单独放在sizeof内部,计算的是第一行数组的大小
printf("%d\n",sizeof(a[0]+1));//4/8
//a[0]不是单独放在sizeof内部,表示的是首元素的地址。即第一行第一个元素的地址,相当于&a[0][0]
//a[0]+1 是第一行第二个元素的地址,相当于&a[0][1]
printf("%d\n",sizeof(a+1));//4/8
//a作为二维数组的数组名,并非单独放在sizeof内部,所以表示首元素是地址
//二维数组的首元素是第一行,这里的a就是第一行的地址
//a+1是跳过第一行,指向第二行
printf("%d\n",sizeof(*(a+1)));//16
//*(a+1) -> a[1],计算的是第二行的大小
printf("%d\n",sizeof(&a[0]+1));//4/8
//&a[0]是第一行的地址
//&a[0]+1是第二行的地址
printf("%d\n",sizeof(*(&a[0]+1)));//表示第二行,大小为16
printf("%d\n",sizeof(*a));//16
//*a --> *(a+0)表示第一行,大小为16
三、总结回顾
- sizeof(数组名),这里的数组名表示整个数组,计算的是数组的大小。
- &数组名,这里的数组名表示整数数组,取出的是整个数组的地址。
- 除此之外所有的数组名都表示首元素的地址。