C指针剖析

指针变量即为存放地址的变量。

指针的基本符号

int* p; p = &a;

在上面的代码中

p为存放地址的变量,其中存放的是指向一个存放于int变量的地址。

*p为p中地址指向的空间

&a为变量a的地址。

注意事项:

1:&a返回的是一个地址。需要强制类型转换为指针变量再赋值给其他变量否则可能会警告,且只能强制类型转换为指针变量否则可能会警告。

2:char *必须对应char变量的地址,int *必须对应int变量的地址。以此类推否则就算不报错也会出bug。

3:如果一个指针暂时没有使用,那么应该赋值为NULL防止指针乱指,称为野指针。

4:创建指针变量的标准是int* p而不是int *p,虽然两种都可以,但是指针变量初始化赋值的时候是赋值给p而非*p;

指针与数组

arr/&arr[0]/&arr

三者关系如下:

arr与&arr[0]都为数组第一个元素的地址。

&arr为整个数组的地址。

在不做运算的情况下三者相等:

arr = &arr[0] = &arr

但是如果做运算

ar++ 等于 &arr[0]++,都是地址偏移一个数据类型的地址空间

&arr++,地址偏移一个数组空间。

例如:如果一个数组为 int a[3];地址为0x...000

那么a++和&a[0]++的值都为0x...000+4 = 0x...004//加4是因为int长度4

而&arr++的值则为0x000+(4 * 3) = 0x00C。//乘3是因为数组长度为3

指针数组与数组指针

int* arr_p[32];//指针数组 int arrb[4] = {1,2,3,4};//创建一个数组,数组指针为arrb

数组指针:数组修饰指针,为普通数组的指针。数组的本质即为连续地址存储的一段数子

指针数组:指针修饰数组,数组中的每个元素都是指针。

int main() 
{     
    int arrb[4] = {1,2,3,4};//创建一个数组
    int *ppp = NULL;//创建一个指针     
    ppp = arrb;//将数组基地址赋值给指针     
    printf("arr_address = %p\n",ppp);//打印结果为数组的地址     
    ppp[3] = 22;//通过指针对数组的值进行修改     
    printf("arr = %d",ppp[3]);//这里打印22,为ppp偏移3后的地址指向的值     
    printf("arr = %p",&ppp[3]);//这里打印的是ppp偏移3个变量大小后的地址     
    return 0; 
}

从上面的例子可以看出[]不是数组专有的而是指针专有的,在指针后加[n]如指针p的p[n],就代表p指针偏移n位指向后指向的数据空间。而&p[n]就代表着p偏移n位后的地址。

而对于数组初始化阶段,[n]意味着申请多少个装入数据类型内存空间,如int arr[n]则申请对应的4(int) * n个字节的空间。

指针的指针

也称为指针链,意思A指针存放着B指针的地址,B指针的地址执行变量。那么A指针就称为二级指针。

多级指针在初始化时前面要加多个*(有多少级就加多少)

int main() 
{     
    int a = 100;     
    int *ap,**p,***bp;//ap为一级指针,p为二级指针,bp为三级指针     
    ap = &a;     
    p = ≈     
    bp = &p;     
    printf("%d\n",***bp);     
    return 0; 
}

注意事项

多级指针在初始化时要加多个*。

函数指针

一个函数的函数名本质是一个指针,是一个函数的入口地址,指向函数的入口。

形式

int (*p)(int) = &mya;//创建一个函数指针(*p)(int)并将其赋值为函数mya的值,&可以省略 p(a);//即为调用函数指针p指向的函数

注意事项:

int (*p)(int)为一个函数指针,开始的int要符合装入的函数,如果装入的函数的返回值为void,那么这里也要一样

int (*p)(int)的第二个括号中要明确其传达的参数。

int mya(int a) 
{     
    return ++a; 
} 
int main() 
{     
    int a = 10;     
    int b = 0;     
    int (*p)(int) = &mya;//创建一个函数指针(*p)(int)并将其赋值为函数mya的值。     
    b = p(a);//这里等价为b = mya(a).     
    printf("a = %d\n",b);//输出结果为11     
    printf("mya_address = %p",mya);//输出结果为函数地址     
    return 0; 
}

回调函数

回调函数即为在一个函数的参数中有其他函数存在。

形式:

一个函数的参数中存在函数指针就是回调函数。

用途:

当使用一个库函数A时,这个函数A中的某些动作需要自己完成,那么就写一个符合的函数从而使得库函数正常运行。当然调用某些API时,不止这一种方法使自己的程序与API吻合,但是C语言在设计时考虑到了这中方法并为之提供了这样的语法用于解决。

例子:

int mya(int a) 
{     
    return ++a; 
} 

int myb(int (*p)(int))//回调函数 
{     
    int m_b = 100;     
    return p(m_b);//在这个函数中调用传参进来的函数,这个动作称为回调 
} 

int main() 
{     
    int mainout = 0;     
    mainout = myb(mya);//调用回调函数并传入一个函数作为回调函数的参数     
    printf("mainout = %d",mainout);//输出结构为101     
    return 0; 
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值