指针应该这样学(C语言)

前言

        相信大家都有耳闻,C语言中的指针是比较难理解的,下面就让我们通过一些简单的练习来深入理解指针。

目录

1.题1

2.题2

3.题3

4.题4

5.题5

6.题6

7.题7

8.题8

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

//程序的结果是什么?

        结果:

分析:&arr是取出数组的整个地址所以,&arr+1实际上是跳过了整个数组。

图解:

2.题2

//这里告知结构体的大小是20个字节
struct Test
{
        int Num;
        char *pcName;
        short sDate;
        char cha[2];
        short sBa[4];
}*p;
//假设p 的值为0x100000。 如下表表达式的值分别为多少?
//已知,结构体Test类型的变量大小是20个字节
int main()
{
        printf("%p\n", p + 0x1);
        printf("%p\n", (unsigned long)p + 0x1);
        printf("%p\n", (unsigned int*)p + 0x1);
        return 0;
}

//程序的结果是什么

        结果:

分析:         不同类型的指针加一所带来的效果不同,注意和指针的类型有关系,整形指针加1,跳过4个字节,结构体类型的指针+1跳过一个结构体的大小,而无符号长整型+1就是数值上的加1。

3.题3

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

//这段代码的结果是什么

        结果:

分析: int *ptr1 = (int *)(&a + 1);这段代码和第一题的意思差不多 ,&a+1,跳过整个数组,而在屏幕上输出的时候用ptr1[-1]相当于*(ptr-1)。

        int *ptr2 = (int *)((int)a + 1);这段代码是将数组a的首地址强制类型转换为整形然后加一,又强制类型转换为int*的指针赋值给ptr2,ptr2解引用就是从这个位置向后访问四个字节的元素。而数组中数据是连续存储的并且是以小端字节序的方式存储。

图解:

4.题4

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

//程序的结果是什么

        结果:

 分析:a是一个二维数组,这个二维数组中的元素是1,0,   2,0,  3,  0。为什么是这个结果呢,因为数组a其实是不完全初始化的,因为数组a初始化时用的是逗号表达式(这个不细心很容易被忽略),逗号表达式的值是括号里面最后一个逗号值的结果。然后将a[0]赋值给p就是将二维数组第一行的首元素赋值给指针p。

5.题5

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

//这段代码的结果是什么

        结果:

分析: 将数组a的首元素的地址也就是第一行的地址赋值给数组指针p,但是p的类型是int[4],a的首元素的类型是int [5],它们两个的类型是有差异的,所以p[4][2]和a[4][2]所在的地址是不一样的。当然打印的结果也就是不一样的。p[4][2]和a[4][2]是两个指针,两个指针相减的到的是数组间的元素个数,由于数组是向上增长的,所以p[4][2]是小于a[4][2]的,所以二者相减得到的是-4,-4,由于在屏幕上面输出的时候以地址的形式输出-4,就会是一个很大的数(因为地址都是非负数)。

图解:

6.题6

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

//这段代码的结果是什么

        结果:

分析:int *ptr1 = (int *)(&aa + 1);&aa+1就是跳过整个二维数组,将&aa+1赋值给ptr1然后向屏幕上输出ptr-1的值输出的就是10。

int *ptr2 = (int *)(*(aa + 1)); aa+1是数组首元素+1,因为aa是二维数组,所以aa的首元素+1其实就是二维数组的第二行的地址然后赋值给ptr2,然后向屏幕中输出ptr2 -1的值就是5。

7.题7

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

//这段代码的结果是什么

        结果:

分析 :a是一个字符指针数组,它里面存放了三个字符串的地址,分别是:"work","at","alibaba",然后将a的首元素的地址给pa,pa自增此时pa中存放的是第二个字符串的地址的地址。*pa,就是第二个字符串的地址,所以在屏幕上输出的是第二个字符串。

8.题8

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

//这段代码的结果是什么

        结果:

分析:这个程序比较复杂需要变看图才能理解,所以我将图和解释放在了一起。如下:

图解:

如果大家还有什么不懂的可以参考C语言指针只看这一篇就够了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值