指针详解(六)

目录

1. sizeof 和 strlen

1)sizeof

2)strlen

3)sizeof 和 strlen的区别

2. 数组和指针试题解析

1)一维数组

2)字符数组

3)二维数组

3. 指针运算试题解析


1. sizeof 和 strlen
1)sizeof

sizeof 计算变量所占内存内存空间大小的,单位是字节,如果操作数是类型的话,计算的是使用类型创建的变量所占内存空间的大小

sizeof 只关注占用内存空间的大小,不在乎内存中存放什么数据

#include <stdio.h>
int main()
{
    int a = 10;
    printf("%d\n", sizeof(a));//int 类型为4字节
    printf("%d\n", sizeof a);
    printf("%d\n", sizeof(int));
    return 0;
}

2)strlen

strlen 是C语言库函数,功能是求字符串长度

size_t strlen ( 1 const char * str );

strlen 函数的参数 str 中首地址开始向后寻找,统计 \0 之前字符串中字符的个数strlen 函数会一直向后寻找 \0 字符,直到找到为止,所以可能存在越界查找

#include <stdio.h>
int main()
{
    char arr1[3] = {'a', 'b', 'c'};
    char arr2[] = "abc";
    printf("%d\n", strlen(arr1));//随机值 
    printf("%d\n", strlen(arr2));// a b c \0 3个元素(字节)
    printf("%d\n", sizeof(arr1));//char * 3 == 3 字节
    printf("%d\n", sizeof(arr2));//a b c \0 4字节
    return 0;
}

 

3)sizeof 和 strlen的区别

sizeof

  1. sizeof 是操作符
  2. sizeof 计算操作数所占内存的大小,单位是字节
  3. 不关注内存中存放什么数据

strlen

  1. strlen是库函数,使用需要包含头文件 string.h
  2. srtlen是求字符串长度的,统计的是 \0 之前字符的元素个数
  3. 关注内存中是否有 \0 ,如果没有\0 ,就会持续往后找,可能会越界访问
     

 

2. 数组和指针试题解析
1)一维数组
int a[] = {1,2,3,4};//4个元素 16字节

printf("%d\n",sizeof(a));// a 为数组名 代表整个数组 int(4字节)*4 == 16
printf("%d\n",sizeof(a+0));// a 为首元素地址 a+0==a ->sizeof(int) == 4(32位)/8(64位)
printf("%d\n",sizeof(*a));// a 为首元素地址 *a==1  sizeof(1) == 4(32位)/8(64位)
printf("%d\n",sizeof(a+1));// a 为首元素地址 a+1 ->2的地址 sizeof(地址)== 4(32位)/8(64位)
printf("%d\n",sizeof(a[1]));// a 为首元素地址 a[i]==*(a+i)==2 sizeof(2)-> 4
printf("%d\n",sizeof(&a));// a 为数组名 &a 为整个数组地址 但还是地址 4(32位)/8(64位)
printf("%d\n",sizeof(*&a));// a 为首元素地址 *&a==a sizeof(a)->int(4字节)*4 == 16
printf("%d\n",sizeof(&a+1));//a 为数组名 &a 为整个数组地址 &a+1 跳过1个数组大小 但还是地址 4(32位)/8(64位)
printf("%d\n",sizeof(&a[0]));//取1的地址 还是地址 ->4(32位)/8(64位)
printf("%d\n",sizeof(&a[0]+1));//取2的地址 还是地址 ->4(32位)/8(64位)

 

2)字符数组
//代码1:
char arr[] = {'a','b','c','d','e','f'};//6个元素(字节)
printf("%d\n", sizeof(arr));// arr 为数组名 sizeof(数组名)->6*1==6(字节)
printf("%d\n", sizeof(arr+0));// arr 为首元素地址 sizeof(首元素地址)-> 地址 == 4(32位)/8(64位)
printf("%d\n", sizeof(*arr));// arr 为首元素地址 *arr == 'a' sizrof('a')-> 1
printf("%d\n", sizeof(arr[1]));// arr 为首元素地址 arr[i]==*(arr+i) ->sizeof('b') == 1
printf("%d\n", sizeof(&arr));// arr 为数组名 &arr 为整个数组地址 但还是地址 -> 4(32位)/8(64位)
printf("%d\n", sizeof(&arr+1));// arr 为数组名 &arr 为整个数组地址 &arr+1 跳过1个数组 但还是地址 -> 4(32位)/8(64位)
printf("%d\n", sizeof(&arr[0]+1));// arr 为首元素地址 &arr[0】为'a'的地址 &arr[0]+1 为'b'的地址 但还是地址 -> 4(32位)/8(64位)

//代码2:
char arr[] = {'a','b','c','d','e','f'};//6个元素(字节)
printf("%d\n", strlen(arr));//随机值 无\0
printf("%d\n", strlen(arr+0));///随机值 无\0
printf("%d\n", strlen(*arr));//err
printf("%d\n", strlen(arr[1]));//err
printf("%d\n", strlen(&arr));//随机值 无\0
printf("%d\n", strlen(&arr+1));//随机值 无\0  跳过1整个数组 &arr的随机值 -6
printf("%d\n", strlen(&arr[0]+1));//随机值 无\0 比 &arr的随机值r -1

//代码3:
char arr[] = "abcdef";//7个元素 a b c d e f \0  7字节
printf("%d\n", sizeof(arr));// arr 为数组名 7*1==7(字节)
printf("%d\n", sizeof(arr+0));// arr 为首元素地址 但还是地址 == 4(32位)/8(64位)
printf("%d\n", sizeof(*arr));//arr 为首元素地址 *arr=='a' sizeof('a') -> 1
printf("%d\n", sizeof(arr[1]));// arr 为首元素地址 arr[i]==*(arr+i) ->sizeof('b') == 1
printf("%d\n", sizeof(&arr));// arr 为数组名 &arr 为整个数组地址 但还是地址 -> 4(32位)/8(64位)
printf("%d\n", sizeof(&arr+1));// arr 为数组名 &arr 为整个数组地址 &arr+1 跳过1个数组 但还是地址 -> 4(32位)/8(64位)
printf("%d\n", sizeof(&arr[0]+1));// arr 为首元素地址 &arr[0】为'a'的地址 &arr[0]+1 为'b'的地址 但还是地址 -> 4(32位)/8(64位)

//代码4:
char arr[] = "abcdef";//7个元素 a b c d e f \0  6字节
printf("%d\n", strlen(arr));// arr 为首元素地址 6
printf("%d\n", strlen(arr+0));// arr 为首元素地址 6
printf("%d\n", strlen(*arr));//err
printf("%d\n", strlen(arr[1]));//err
printf("%d\n", strlen(&arr));//arr 为数组名 &arr 为首元素地址 6
printf("%d\n", strlen(&arr+1));//随机值
printf("%d\n", strlen(&arr[0]+1));// arr 为首元素地址 arr[i]==*(arr+i) ->从第二个‘b'开始计数 
 -> 6-1 == 5

//代码5:
char *p = "abcdef";//p == &a
printf("%d\n", sizeof(p));//p为地址 == 4(32位)/8(64位)
printf("%d\n", sizeof(p+1));// 还是地址 == 4(32位)/8(64位)
printf("%d\n", sizeof(*p));//sizeof('a')  1
printf("%d\n", sizeof(p[0]));//sizeof('a')  1
printf("%d\n", sizeof(&p));//还是地址 == 4(32位)/8(64位)
printf("%d\n", sizeof(&p+1));//还是地址 == 4(32位)/8(64位)
printf("%d\n", sizeof(&p[0]+1));//还是地址 == 4(32位)/8(64位)

//代码6:
char *p = "abcdef";//p == &a
printf("%d\n", strlen(p));//从 a 开始计数 6 
printf("%d\n", strlen(p+1));//从 b 开始计数 5
printf("%d\n", strlen(*p));//err
printf("%d\n", strlen(p[0]));//err
printf("%d\n", strlen(&p));//随机值 二级指针
printf("%d\n", strlen(&p+1));//随机值 比&p得随机数小1
printf("%d\n", strlen(&p[0]+1));//从 b 开始计数 5

 

 

 

 

 

3)二维数组
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]));

数组名的意义:

  • sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小
  • &数组名,这里的数组名表示整个数组,取出的是整个数组的地址
  • 除此之外所有的数组名都表示首元素的地址

 

3. 指针运算试题解析
//题目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));//2,5
    return 0;
}

//题目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
#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;
}

//题目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;
}

//题目5
#include <stdio.h>
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;
}

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

//题目7
#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;
}

谢谢观看

  • 7
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值