【C语言】进阶指针,超详解,含丰富代码示例

文章目录

  • 前言
  • 指针进阶的重点内容
    • 1.字符指针
    • 2.数组指针
    • 3.指针数组
    • 4.函数指针
    • 5.函数指针数组
    • 6. 指向函数指针数组的指针
  • 总结


这里是初阶的链接,方便大家对照查看!!!添加链接描述

前言

大家好呀,今天和大家将指针进阶的知识进行分享,这块是指针的难点部分,希望博主对其的理解可以帮助到大家!!!


指针进阶的重点内容

  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 里了,但是/本质是把字符串 hello bit. 首字符的地址放到了pstr中。
并且,这里的"hello bit."字符串是常量字符串不允许对其进行修改

下面介绍有关的一道经典面试题:

#include <stdio.h>
int main()
{char str1[] = "hello bit.";char str2[] = "hello bit.";char* str3 = "hello bit.";char* str4 = "hello bit.";if (str1 == str2)printf("str1 and str2 are same\n");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("str3 and str4 are not same\n");return 0;
}

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

2.数组指针

整形指针: int * pint; 能够指向整形数据的指针。
浮点型指针: float * pf; 能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针

下面就是一个数组指针的定义:

int (*p)[10];
//解释:p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个
指针,指向一个数组,叫数组指针。
//这里要注意:[]的优先级要高于*号的,所以必须加上()来保证p先和*结合。

知道了数组指针,那么我们就要了解一下 数组名和&数组名 的含义了:

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

在这里插入图片描述
&arr 表示的是数组的地址,而不是数组首元素的地址。本例中 &arr 的类型是: int(*)[10] ,是一种数组指针类型数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40.

3.指针数组

指针数组是指针还是数组?
:是数组。是存放指针的数组。
可以用一个指针数组来模拟一个二维数组,代码如下:

int main()
{int arr1[10] = { 1,5,6,8,2,36,9,5 };int arr2[10]={55,6,8,46,89};int arr3[10] = {55,659,1022,651,365};int i = 0, j = 0;//用指针数组描述二位数组int* arr[3] = {arr1,arr2,arr3};for (i = 0; i < 3; i++){for (j = 0; j < 10; j++){printf("%d ",arr[i][j]);}printf("\n");}return 0;
}

在这里插入图片描述

4.函数指针

大家首先看一段代码:

#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");
}
void (*pfun1)();
void *pfun2();

//上面pfun1和pfun2哪个有能力存放test函数的地址?

pfun1可以存放。pfun1先和*结合,说明pfun1是指针,指针指向的是一个函数,指向的函数无参数,返回值类型为void。
根据这个定义的方法我们就可以定义除任何函数的函数指针类型变量了。

//代码1
(*(void (*)())0)();
//代码2
void (*signal(int , void(*)(int)))(int);
//学会函数指针后,大家可以尝试分析一下这两个代码的含义,博主会在评论区揭晓答案!!!

5.函数指针数组

那要把函数的地址存到一个数组中,那这个数组就叫函数指针数组。

int (*parr1[10])();

parr1 先和 [] 结合,说明 parr1是数组,数组的内容是什么呢?是 int (*)() 类型的函数指针
应用举例:

#include<stdio.h>
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}
void menu()
{printf("******************************\n");printf("****1、加法   2、减法   ******\n");printf("****3、乘法   4、除法   ******\n");printf("***********0、退出************\n");
}
int main()
{//利用了函数指针数组int (*Cal[5])(int, int) = { 0,Add,Sub,Mul,Div };int input = 0;do{menu();printf("请输入你的选择:\n");scanf("%d", &input);if (input == 0){printf("退出计算器!\n");}else if (input >= 1 && input <= 4){int x = 0, y = 0;printf("请输入计算的数据:\n");scanf("%d %d", &x, &y);int ret = (*(Cal[input]))(x, y);//int ret=Cal[input](x,y);//和上面一行意思相同printf("结果为:%d\n", ret);}else{printf("选择错误,重新输入!\n");}} while (input);
}

在这里插入图片描述
这个例子就是对函数指针数组应用的很好举例,将加减乘除四个函数保存在函数指针数组中进行调用,极大程度减少了代码的繁琐程度

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

指向函数指针数组的指针是一个 指针
指针指向一个 数组 ,数组的元素都是 函数指针 ;
如何定义?
请看下面代码展示:

void test(const char* str)
{printf("%s\n", str);
}
int main()
{//函数指针pfunvoid (*pfun)(const char*) = test;//函数指针的数组pfunArrvoid (*pfunArr[5])(const char* str);pfunArr[0] = test;//指向函数指针数组pfunArr的指针ppfunArrvoid (*(*ppfunArr)[5])(const char*) = &pfunArr;//ppfunArr是一个指针,它指向一个有五个元素的数组,数组每个元素是函数指针类型!return 0;
}

总结

本次指针进阶的内容就先为大家分享到这里,后面还有一个比较重要的回调函数的使用技巧,包括使用qsort函数及其实现我们自己的一个qsort函数博主会放在紧接着的下一篇博客中详细解说,希望本片文章可以帮助到大家,博主有讲解说明不到位的地方还恳请大家指正,谢谢大家,我们下期见!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/63846.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

运维监控学习笔记1

1、监控对象&#xff1a; 1、监控对象的理解&#xff1b;CPU是怎么工作的&#xff1b; 2、监控对象的指标&#xff1a;CPU使用率&#xff1b;上下文切换&#xff1b; 3、确定性能基准线&#xff1a;CPU负载多少才算高&#xff1b; 2、监控范围&#xff1a; 1、硬件监控&#x…

JVM内存区域

预备 为了更好的理解类加载和垃圾回收&#xff0c;先要了解一下JVM的内存区域&#xff08;如果没有特殊说明&#xff0c;都是针对的是 HotSpot 虚拟机。&#xff09;。 Java 源代码文件经过编译器编译后生成字节码文件&#xff0c;然后交给 JVM 的类加载器&#xff0c;加载完…

SpringBoot复习:(36)国际化

一、Resources目录下建立一个目录&#xff08;比如international)来存储资源文件 message.properties 空的&#xff0c;但不能没有 message_zh_CN.properties hello您好message_en_us.properties hellohello world二、自动配置类MessageSourceAutoConfiguration 常量MESSAGE…

【MongoDB】解决ProxmoxVE下CentOS7虚拟机安装MongoDB6后启动失败的问题

目录 安装步骤&#xff1a; 2.1 配置yum源 2.2 安装MongoDB 2.3 启 动MongoDB ProxmoxVE上新装的CentOS7.4虚拟机&#xff0c;安装MongoDB6。 安装步骤&#xff1a; 2.1 配置yum源 # 创建mongodb yum源&#xff08;https://www.mongodb.com/docs/manual/tutorial/insta…

RTT(RT-Thread)线程间同步(保姆级)

目录 线程间同步 信号量 信号量结构体 信号量的使用和管理 动态创建信号量 实例 静态创建信号量 初始化和脱离信号量 获取信号量 信号量的互斥操作 获取信号量函数 释放信号量 信号量同步实例 互斥量&#xff08;互斥锁&#xff09; 互斥量的使用和管理 动态创…

【idea】点击idea启动没反应

RT 点击idea启动的时候没反应&#xff0c;接着百度报错&#xff0c;基本跟他们的也不一样。 首先我是做版本升级。其次&#xff0c;我之前是破解的。如果你也是跟我一样的话&#xff0c;那问题可能就处在破解上了 解决方式 首先&#xff0c;是跟大部分解决思路一样。先找到项…

AWS中Lambda集成SNS

1.创建Lambda 在Lambda中&#xff0c;创建名为AWSSNSDemo的函数 use strict console.log(loading function); var aws require(aws-sdk); var docClient new aws.DynamoDB.DocumentClient(); aws.config.regionap-southeast-1;exports.handler function(event,context,cal…

【Linux 网络】 数据链路层协议

数据链路层协议 数据链路层解决的问题以太网协议认识以太网以太网帧格式 认识MAC地址对比理解MAC地址和IP地址认识MTUMTU对IP协议的影响MTU对UDP协议的影响MTU对于TCP协议的影响ARP协议ARP协议的作用ARP协议的工作流程ARP数据报的格式 总结 数据链路层解决的问题 IP拥有将数据跨…

【CSS】CSS 布局——常规流布局

<h1>基础文档流</h1><p>我是一个基本的块级元素。我的相邻块级元素在我的下方另起一行。</p><p>默认情况下&#xff0c;我们会占据父元素 100%的宽度&#xff0c;并且我们的高度与我们的子元素内容一样高。我们的总宽度和高度是我们的内容 内边距…

python爬虫——爬虫伪装和反“反爬”

前言 爬虫伪装和反“反爬”是在爬虫领域中非常重要的话题。伪装可以让你的爬虫看起来更像普通的浏览器或者应用程序&#xff0c;从而减少被服务器封禁的风险&#xff1b;反“反爬”则是应对服务器加强的反爬虫机制。下面将详细介绍一些常见的伪装和反反爬技巧&#xff0c;并提…

ts中interface自定义结构约束和对类的约束

一、interface自定义结构约束对后端接口返回数据 // interface自定义结构 一般用于较复杂的结构数据类型限制 如后端返回的接口数据// 首字母大写;用分割号隔开 interface Iobj{a:number;b:string } let obj:Iobj {a:1,b:2 }// 复杂类型 模拟后端返回的接口数据 interface Il…

泰国的区块链和NFT市场调研

泰国的区块链和NFT市场调研 基本介绍 参考&#xff1a; https://zh.wikipedia.org/zh-hans/%E6%B3%B0%E5%9B%BD参考&#xff1a; https://hktdc.infogram.com/thsc–1h7k2303zo75v2x zz制度&#xff1a; 君主立宪制&#xff08;议会制&#xff09; 国王&#xff1a; 玛哈哇集拉…