当前位置: 首页 > article >正文

关于我、重生到500年前凭借C语言改变世界科技vlog.13——深入理解指针(3)

文章目录

  • 1.字符指针变量
  • 2.数组指针变量
  • 3.函数指针变量
  • 4.函数指针数组
  • 5.二维数组传参本质
  • 6.拓展补充
  • 希望读者们多多三连支持
  • 小编会继续更新
  • 你们的鼓励就是我前进的动力!

本章节接着学习常见的指针变量类型

1.字符指针变量

字符指针变量,顾名思义就是字符类型的指针,即 char*
常见的输出格式是这样的:

int main()
{
 char ch = 'w';
 char *pc = &ch;
 *pc = 'w';
 return 0;
}

这是存放一个字符的情况,如果存放字符串呢?

int main()
{
 const char* pstr = "hello bit.";
 printf("%s\n", pstr);
 return 0;
}

乍一看是存放字符串在指针变量中,但我们要记住指针变量是用来存放地址的
所以这里本质是把字符串 hello bit. 首字符的地址放到了pstr中,即字符 h 的地址

2.数组指针变量

上一篇 vlog 学到了指针数组,就是存放指针的数组,也可以理解为存放指针的集合(元素相同),那么数组指针就可以得出是存放数组地址的指针,是一种指针变量,指向数组

int *p1[10];
int (*p2)[10];

那么以上哪种是数组指针?
答案是下面那个
分析:[ ] 的优先级大于 * ,所以必须加上 [ ] 来保证 p 和 * 优先结合

p先和 * 结合,说明p是一个指针变量,然后指针指向的是一个大小为10个整型的数组
所以 p 是一个指针,指向一个数组,叫数组指针

int 表示 p指向的数组的元素类型, p 是数组指针变量名,10是指向数组的元素个数

3.函数指针变量

根据前面学过的类比,不难发现,函数指针变量应该是用来存放函数地址的,通过地址能够调用函数的
那么函数真的有地址吗?

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

通过以上代码可以发现函数确实有地址,用函数名就能代表其地址,当然也可以通过 &函数名 的方
式获得函数的地址,为了方便一般就不写取地址符
其语法形式为:

int(*pf3)(int, int) = Add;
int(*pf3)(int x, int y) = &Add;

函数参数的变量名可写可不写,取地址符也是
int 是指向函数的返回类型,pf3 是函数指针变量名,int x,int y 是 pf3 指向函数的参数类型和个数

#include <stdio.h>
int Add(int x, int y)
{
 return x+y;
}
int main()
{
 int(*pf3)(int, int) = Add;
 
 printf("%d\n", (*pf3)(2, 3));
 printf("%d\n", pf3(3, 5));
 return 0;
}

可以将通过函数指针调用指针指向的函数写一个我们之前写过的加法函数

这里通过解引用函数指针 pf3 的方式来调用它所指向的函数(也就是 Add 函数),传入参数 2 和 3,然后将返回的结果使用 printf
函数输出。实际上,在这种情况下,解引用操作符 * 在这里是可选的,因为在 C
语言中,函数名本身在求值时就会转换为指向该函数的指针,所以也可以直接写成 pf3(2, 3)

4.函数指针数组

在学习了指针数组的基础上,我们引入函数指针放入数组
那么以下哪种为正确的形式?

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

答案是第一个

定义形式如下:返回值类型 (*数组名[数组大小])(参数列表)

parr1 先和 [ ] 结合,说明 parr1是数组,是 int (*)() 类型的函数指针
那么参数如何理解?其实就是每个元素代表的函数

#include <stdio.h>

int add(int a, int b) {
    return a + b;
}

int subtract(int a, int b) {
    return a - b;
}

int main() {
    int (*func_array[2])(int, int) = {add, subtract};
    return 0;
}

在上述代码中,func_array 数组的两个元素分别被初始化为 add 函数和 subtract 函数的指针

5.二维数组传参本质

讲数组的时候说过二维数组其实可以看做是每个元素是一维数组的数组,也就是二维数组的每个元素是一个一维数组,那么二维数组的首元素就是第一行,是个一维数组
在这里插入图片描述
第一行的一维数组的类型就是 int [5] ,所以第一行的地址的类型就是数组指针类型 int(*)[5] ,那就意味着二维数组传参本质上也是传递了地址,传递的是第一行这个一维数组的地址,那么形参也是可以写成指针形式的,总的来说就是把二维数组当一维数组理解,第一行看成一维数组的第一个元素,首元素就是第一行一整行的地址

#include <stdio.h>

 void test(int (*p)[5], int r, int c)
 {
 int i = 0;
 int j = 0;
 for(i=0; i<r; i++)
 {
 for(j=0; j<c; j++)
 {
 printf("%d ", *(*(p+i)+j));
 }
 printf("\n");
 }
}
int main()
{
 int arr[3][5] = {{1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7}};
 test(arr, 3, 5);
 return 0;
}

普通的遍历数组传参也可以这样写

这里的 *(p+i) 相当于获取二维数组的第 i 行的首地址(因为 p 是指向包含 5 个整数的数组的指针,p+i 就指向了第 i 行),然后 *(p+i)+j 就是指向第 i 行第 j 列元素的指针,最后 ((p+i)+j) 就是获取该位置的元素值并输出

虽然解引用通常是获取元素本身,但在指向二维数组行的指针这种特殊情况下,由于指针所指向的对象本身就是一个数组,解引用得到的就是这个数组的首地址,这是由 C 语言的指针和数组特性共同决定的

二维数组传参,形参的部分可以写成数组,也可以写成指针形式

6.拓展补充

补充一个关键字 typedef ,是用来类型重命名的,可以将复杂的类型,简单化

普通类型

typedef unsigned int uint;
//将unsigned int 重命名为uint

普通指针类型

typedef int* ptr;

数组函数指针类型

typedef int(*parr)[5]; //新的类型名必须在*的右边
typedef void(*pfun)(int);//新的类型名必须在*的右边

下一期 vlog 将对二分查找,转移表,冒泡排序等常见算法题目进行练习解析
建议对前面的知识都有系统性的理解后再来写题

主页传送门:DARLING Zero two♡ 的 blog

希望读者们多多三连支持

小编会继续更新

你们的鼓励就是我前进的动力!

在这里插入图片描述


http://www.kler.cn/a/374654.html

相关文章:

  • T4--侯豆病毒识别
  • jenkins ssh 免密报错Host key verification failed.
  • 使用 Python 的 BeautifulSoup 与 Flask/Flask-RESTful 集成进行数据爬取和 API 构建
  • Golang | Leetcode Golang题解之第516题最长回文子序列
  • 学Linux的第六天
  • 《神经网络助力战场车辆及部件损毁识别与评估》
  • Glide 简易教程
  • 【Rust标准库中的convert(AsRef,From,Into,TryFrom,TryInto)】
  • PyQt5信号与槽一
  • 【抽代复习笔记】34-群(二十八):不变子群的几道例题
  • .net core中间件Polly
  • 【WPF】如何获取屏幕比例
  • BFH的原理及用法
  • 【VS中Git同步提交 报错:访问.vs/FileContentIndex/xxx.vsidx权限不允许】
  • DAO模式及单例模式
  • 查询引擎的演变之旅 | OceanBase原理解读
  • 2024 Rust现代实用教程 Borrowing借用 Lifetime生命周期
  • python将mongodb中的数据写入到postgresql中
  • 华为实时视频使用FLV播放RTSP流
  • ssm药店管理系统-计算机毕业设计源码81276
  • 【数据结构与算法】第7课—数据结构之队列
  • 超子物联网HAL库笔记:准备篇
  • Hive的数据存储格式
  • 设计模式 策略模式 场景Vue (技术提升)
  • WebMvcConfigurer
  • React 中useState 原理