c语言指针进阶

目录

  1. 1.字符指针
  2. 2.指针数组
  3. 3.数组指针
  4.      3.1 数组指针的定义

         3.2 &数组名VS数组名

         3.3 数组指针的使用

  5. 4. 数组传参和指针传参
  6.      4.1 一维数组传参

         4.2 二维数组传参

         4.3 一级指针传参

         4.4 二级指针传参

  7. 5. 函数指针
  8. 6. 函数指针数组
  9. 7. 指向函数指针数组的指针
  10. 8. 回调函数

  11. 1.字符指针

  1. 回顾
  2. 1.什么是指针变量
  3. 内存会化分为字节为单位的空间,每一个字节都有一个编号(地址/指针),指针/地址要被存起来,需要一个空间,这个空间就是指针变量

2.指针的大小

指针的大小是固定的4/8个字节(32位平台/64为平台)

3.32位平/64位平台怎样理解

电脑硬件(主要指cpu)

32位虚拟地址—》32bit位的地址

64位虚拟地址—》64bit位的地址

4. 指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限
步长:一个in类型t是4个字节 char类型是一个字节 

权限:一个int*类型解引用只能访问4个字节空间,一个char*类型解引用只能访问1个字节空间

5.指针的运算

那现在开始学习字符指针

字符指针我们应用一般有两种写法

第一种:

char ch = "w"
char *p = &ch

第二种:

char *p = "abcdef"

第一种和第二种的内存布局是什么样子的

第一种创建一个字符变量叫ch ch里面放一个字符w 字符变量有自己的地址 我们把字符变量地址取出来放在 p里面去 p是一个指针变量它里面存的是ch的地址

第二种 char* = “abcdef”中的“abcdef”是存在只读数据区的,它存在内存中也有他的地址而它的地址是“abcdef”中的起始地址是a的地址 我们是把a的地址存在p里面

一道例题

include <stdio.h>
int main()
{
char str1[] = "abcdef";
char str2[] = "abcdef";
const char *str3 = "abcdef";
const char *str4 = "abcdef";
if(str1 ==str2)
printf("str1 == str2\n");
else
printf("str1 != str2 \n");
if(str3 ==str4)
printf("str3 == str4 \n");
else
printf("str3 != str4 \n");
return 0;
}

运行结果

 因为str1和str2是两块不同的数组,而数组名代表首元素的首地址 str1和str2是两块不同的地址,而char*str3和char*str4里存的是常量字符串 常量字符串不能修改

这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当
几个指针。指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化
不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4不同


2.指针数组

指针数组是一个存放指针的数组

例子

int *arr1[10]

arr1首先和方括号结合变成数组了它有10个元素 每个元素类型是int*

char *arr2[4]

arr2和方括号结合变成数组,它有4个元素,每个元素类型是char*

char **arr3[5]

arr3和方括号结合成数组了,它有5个元素,每个元素类型是char**

3.数组指针

   3.1数组指针的定义

数组指针是指针?还是数组?
答案是:指针。
我们已经熟悉:
整形指针: int * pint; 能够指向整形数据的指针。
浮点型指针: float * pf; 能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针。
下面代码哪个是数组指针?
int *p1[10];
int (*p2)[10];
//p1, p2分别是什么?
解释
int (*p)[10];
//解释:p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个
指针,指向一个数组,叫数组指针。
//这里要注意:[]的优先级要高于*号的,所以必须加上()来保证p先和*结合。
 

3.2 &数组名VS数组名
对于下面的数组:

int arr[10];

arr 和 &arr 分别是啥?
我们知道arr是数组名,数组名表示数组首元素的地址。
那&arr数组名到底是啥?

我们看一段代码:

#include <stdio.h>
int main()
{
int arr[10] = {0};
printf("%p\n", arr);
printf("%p\n", &arr);
return 0;
}

根据上面的代码我们发现,其实&arr和arr,虽然值是一样的,但是意义应该不一样的。
实际上: &arr 表示的是数组的地址,而不是数组首元素的地址。(细细体会一下)
本例中 &arr 的类型是: int(*)[10] ,是一种数组指针类型
数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40
 3.3 数组指针的使用

那数组指针是怎么使用的呢?
既然数组指针指向的是数组,那数组指针中存放的应该是数组的地址。
看代码:
 

#include <stdio.h>
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,0};
int (*p)[10] = &arr;//把数组arr的地址赋值给数组指针变量p
//但是我们一般很少这样写代码
return 0;
}

一个数组指针的使用:

#include <stdio.h>
void print_arr1(int arr[3][5], int row, int col)
{
int i = 0;
for(i=0; i<row; i++)
{
for(j=0; j<col; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
void print_arr2(int (*arr)[5], int row, int col)
{
int i = 0;
for(i=0; i<row; i++)
{
for(j=0; j<col; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
int main()
{
int arr[3][5] = {1,2,3,4,5,6,7,8,9,10};
print_arr1(arr, 3, 5);
//数组名arr,表示首元素的地址
//但是二维数组的首元素是二维数组的第一行
//所以这里传递的arr,其实相当于第一行的地址,是一维数组的地址
//可以数组指针来接收
print_arr2(arr, 3, 5);
return 0;
}

学了指针数组和数组指针我们来一起回顾并看看下面代码的意思:

int arr [ 5 ];   //数组
int * parr1 [ 10 ];  //指针数组
int ( * parr2 )[ 10 ];  //数组指针
int ( * parr3 [ 10 ])[ 5 ];  //存放数组指针的数组

4.数组传参和指针传参

     4.1 一维数组传参

#include <stdio.h>
void test(int arr[])//ok? 可以 传上来的是数组 可以有数组接受 本质上是就是指针
{}
void test(int arr[10])//ok? 可以
{}
void test(int *arr)//ok?  可以 传上来的是数组的首元素地址可以用指针来接收
{}
void test2(int *arr[20])//ok? 可以
{}
void test2(int **arr)//ok? 可以 传上来的是arr2一维数组的地址可以用二级指针来接收
{}
int main()
{
int arr[10] = {0}; 整形传参
int *arr2[20] = {0}; 整形数组指针传参
test(arr);
test2(arr2);
}

     4.2 二维数组传参

void test(int arr[3][5])//ok? 可以
{}
void test(int arr[][])//ok? 不可以 第二个必须传参
{}
void test(int arr[][5])//ok? 可以
{}
//总结:二维数组传参,函数形参的设计只能省略第一个[]的数字。
//因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。
//这样才方便运算。
void test(int *arr)//ok?不可以 类型不同  
{}
void test(int* arr[5])//ok? 不可以 类型不同
{}
void test(int (*arr)[5])//ok?可以
{}
void test(int **arr)//ok?不可以
{}
int main()
{
int arr[3][5] = {0};
test(arr);
}

  4.3 一级指针传参

#include <stdio.h>
void print(int *p, int sz)
{
int i = 0;
for(i=0; i<sz; i++)
{
printf("%d\n", *(p+i));
}
}
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9};
int *p = arr;
int sz = sizeof(arr)/sizeof(arr[0]);
//一级指针p,传给函数
print(p, sz);
return 0;
}

  4.4 二级指针传参

 #include <stdio.h>
void test(int** ptr)
{
printf("num = %d\n", **ptr);
}

int main()
{
int n = 10;
int*p = &n;
int **pp = &p;
test(pp);
test(&p);
return 0;
}
思考:
当函数的参数为二级指针的时候,可以接收什么参数?

void test(char **p)
{ }
int main()
{
char c = 'b';
char*pc = &c;
char**ppc = &pc;
char* arr[10];
test(&pc);
test(ppc);
test(arr);//Ok?
return 0;
}

5.函数指针

首先看一段代码:

#include <stdio.h>
void test()
{
printf("hehe\n");
}
int main()
{
printf("%p\n", test);
printf("%p\n", &test);
return 0;
}

输出的结果:

 输出的是两个地址,这两个地址是 test 函数的地址。
那我们的函数的地址要想保存起来,怎么保存?
下面我们看代码:

void test()
{
printf("hehe\n");
}
//下面pfun1和pfun2哪个有能力存放test函数的地址?
void (*pfun1)();
void *pfun2();

首先,能给存储地址,就要求pfun1或者pfun2是指针,那哪个是指针?
答案是:
pfun1可以存放。pfun1先和*结合,说明pfun1是指针,指针指向的是一个函数,指向的函数无参
数,返回值类型为void
 

阅读两段有趣的代码:

//代码1
(*(void (*)())0)();

void (*)() -> 函数指针类型. 0前面有个括号,(void (*)())0,把0进行强制类型转换,此时0就是一个函数地址,此时再加个*进行解引用,再加一个括号去调用函数,不需要传参,因为指向的函数是无参的
把0进行强制转换成函数指针类型,该指针指向的函数是无参,返回类型是void,当0变成一个函数地址时,对它进行解引用操作,去调用以0为地址的该函数。

//代码2
void (*signal(int , void(*)(int)))(int);

  1. signal是一个函数声明,
  2. signal函数的参数有2个,第一个是int,第二个是函数指针,该函数指针指向的函数的参数是int,返回类型是void
  3. signal函数的返回类型也是一个函数指针,该函数指针指向的函数的参数是int,返回类型是void

代码2太复杂,如何简化:

typedef void(*pfun_t)(int);
pfun_t signal(int, pfun_t);
 

6. 函数指针数组

数组是一个存放相同类型数据的存储空间,那我们已经学习了指针数组,
比如

int *arr[10];
//数组的每个元素是int*
 

那要把函数的地址存到一个数组中,那这个数组就叫函数指针数组,那函数指针的数组如何定义呢?

int (*parr1[10])();
int *parr2[10]();
int (*)() parr3[10];
 

答案是:parr1
parr1 先和 [] 结合,说明 parr1是数组,数组的内容是什么呢?
是 int (*)() 类型的函数指针。
函数指针数组的用途:转移表

例子:(计算器)

#include <stdio.h>
int add(int a, int b)
{
return a + b;
}
int sub(int a, int b)
{
return a - b;
}
int mul(int a, int b)
{
return a*b;
}
int div(int a, int b)
{
return a / b;
}
int main()
{
int x, y;
int input = 1;
int ret = 0;
do
{
printf( "*************************\n" );
printf( " 1:add 2:sub \n" );
printf( " 3:mul 4:div \n" );
printf( "*************************\n" );
printf( "请选择:" );
scanf( "%d", &input);
switch (input)
{
case 1:
printf( "输入操作数:" );
scanf( "%d %d", &x, &y);
ret = add(x, y);
printf( "ret = %d\n", ret);
break;
case 2:
printf( "输入操作数:" );
scanf( "%d %d", &x, &y);
ret = sub(x, y);
printf( "ret = %d\n", ret);
break;
case 3:
printf( "输入操作数:" );
scanf( "%d %d", &x, &y);
ret = mul(x, y);
printf( "ret = %d\n", ret);
break;
case 4:
printf( "输入操作数:" );
scanf( "%d %d", &x, &y);
ret = div(x, y);
printf( "ret = %d\n", ret);
break;
case 0:
printf("退出程序\n");
breark;
default:
printf( "选择错误\n" );
break;
}
} while (input);
return 0;

使用函数指针数组的实现:

#include <stdio.h>
int add(int a, int b)
{
return a + b;
}
int sub(int a, int b)
{
return a - b;
}
int mul(int a, int b)
{
return a*b;
}
int div(int a, int b)
{
return a / b;
}
int main()
{
int x, y;
int input = 1;
int ret = 0;
int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //转移表
while (input)
{
printf( "*************************\n" );
printf( " 1:add 2:sub \n" );
printf( " 3:mul 4:div \n" );
printf( "*************************\n" );
printf( "请选择:" );
scanf( "%d", &input);
if ((input <= 4 && input >= 1))
{
printf( "输入操作数:" );
scanf( "%d %d", &x, &y);
ret = (*p[input])(x, y);
}
else
printf( "输入有误\n" );
printf( "ret = %d\n", ret);
}
return 0;
}


7.指向函数指针数组的指针

指向函数指针数组的指针是一个 指针
指针指向一个 数组 ,数组的元素都是 函数指针 ;
如何定义?
 

void test(const char* str)
{
printf("%s\n", str);
}
int main()
{
//函数指针pfun
void (*pfun)(const char*) = test;
//函数指针的数组pfunArr
void (*pfunArr[5])(const char* str);
pfunArr[0] = test;
//指向函数指针数组pfunArr的指针ppfunArr
void (*(*ppfunArr)[5])(const char*) = &pfunArr;
return 0;
}

8.回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个
函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数
的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进
行响应

 首先演示一下qsort函数的使用

#include <stdio.h>
//qosrt函数的使用者得实现一个比较函数
int int_cmp(const void * p1, const void * p2)
{
return (*( int *)p1 - *(int *) p2);
}
int main()
{
int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
int i = 0;
qsort(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);
for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++)
{
printf( "%d ", arr[i]);
}
printf("\n");
return 0;
}

使用回调函数,模拟实现qsort(采用冒泡的方式)。
注意:这里第一次使用 void* 的指针,讲解 void* 的作用。
 

#include <stdio.h>
int int_cmp(const void * p1, const void * p2)
{
return (*( int *)p1 - *(int *) p2);
}
void _swap(void *p1, void * p2, int size)
{
int i = 0;
for (i = 0; i< size; i++)
{
char tmp = *((char *)p1 + i);
*(( char *)p1 + i) = *((char *) p2 + i);
*(( char *)p2 + i) = tmp;
}
}
void bubble(void *base, int count , int size, int(*cmp )(void *, void *))
{
int i = 0;
int j = 0;
for (i = 0; i< count - 1; i++)
{
      for (j = 0; j<count-i-1; j++)
      {
            if (cmp ((char *) base + j*size , (char *)base + (j + 1)*size) > 0)
            {
                       _swap(( char *)base + j*size, (char *)base + (j + 1)*size, size);
            }
       }

 }
}
int main()
{
int arr[] = { 1, 3, 5, 7, 9, 2, 4, 6, 8, 0 };
//char *arr[] = {"aaaa","dddd","cccc","bbbb"};
int i = 0;
bubble(arr, sizeof(arr) / sizeof(arr[0]), sizeof (int), int_cmp);
for (i = 0; i< sizeof(arr) / sizeof(arr[0]); i++)
{
printf( "%d ", arr[i]);
}
printf("\n");
return 0;
}
 

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值