【C语言】——指针四:字符指针与函数指针变量

【C语言】——指针四:字符指针与函数指针变量

    • 一、字符指针
    • 二、函数指针变量
      • 2.1、 函数指针变量的创建
      • 2.2、两段有趣的代码
    • 三、typedef关键字
      • 3.1、typedef的使用
      • 3.2、typedef与define比较
    • 四、函数指针数组

一、字符指针

  在前面的学习中,我们知道有一种指针类型为字符指针: c h a r ∗ char* char。下面我们来介绍它的使用方法。
  
使用方法:

#include<stdio.h>int main()
{char ch = 'w';char* pc = &ch;*pc = 'w';return 0;
}

  
  如果我们想存储字符串,可以用什么方法呢?之前我们一般都是用字符数组,那还有什么办法呢?其实,字符指针也是可以的。
  
使用方法:

#include<stdio.h>int main()
{const char* pstr = "hello world";printf("%s\n", pstr);return 0;
}

  在const char* pstr = "hello world";代码中,可能很多小伙伴以为是把整个字符串"hello world"放进字符指针 p s t r pstr pstr 中,但其实,这里本质是把 " h e l l o "hello "hello w o r l d " world" world"首字符 ‘ h ’ ‘h’ h 的地址放在指针变量 p s t r pstr pstr 中。
  
  至于代码printf("%s\n", pstr);指针 p s t r pstr pstr不需要解引用,因为 p r i n t f printf printf 函数打印字符串本质是接收该字符串首元素地址,从该地址开始往后打印,直到遇到 ‘ \0 ’ 停止,解引用反而是错的。
  
  这里,也要随便提一下,代码printf("hello world"),同样不是把整个字符串 " h e l l o "hello "hello w o r l d " world" world" 传给 p r i n t f printf printf 函数,其本质也是将首字符 ‘ h ’ ‘h’ h 的地址传给 p r i n t f printf printf 。  p r i n t f printf printf 再从给来的地址开始打印,直到遇到 ‘ \0 ’ 停下。
  

在这里插入图片描述

  
  那字符指针与数组指针有什么区别呢?他们最大的区别就是

  • 字符数组里的内容可以修改
  • 字符指针中放的是常量字符串,内容不可修改
      
    因此,我们可以在字符指针 c h a r char char*前加上 c o n s t const const 修饰,以确保他不能被修改。
        
      
    下面,我们来看一道题,进一步感受字符指针与字符数组的区别
int main()
{char str1[] = "hello world";char str2[] = "hello world";const char* str3 = "hello world";const char* str4 = "hello world";if (str1 == str2){printf("str1 and str2 are same\n");}else{printf("str1 and str2 are not same\n");}if (str3 == str4){printf("str3 and str4 are same\n");}else{printf("str3 and str4 are not same\n");}return 0;
}

输出结果:
在这里插入图片描述
  

为什么会这样呢?
  
  这里,其实 s t r 3 str3 str3 s t r 4 str4 str4 指向同一个常量字符串,C/C++会把常量字符串存储到单独的一个内存空间(代码段)。
  
  因为常量字符串无法被修改,没必要存储两份,当多个字符指针指向同一个常量字符串时,他们实际会指向同一块内存
  
  但是用相同的常量字符串去初始化数组就会开辟出不同的内存块
  
  所以 s t r 1 str1 str1 s t r 2 str2 str2 不同, s t r 3 str3 str3 s t r 4 str4 str4 相同。
  
  

二、函数指针变量

2.1、 函数指针变量的创建

  
  什么是函数指针变量呢?
  
  在前面的学习中(【C语言】—— 指针三 : 参透数组传参的本质)我们了解到数组指针变量,他是用来存放数组指针地址的。同理函数指针变量应该是存放函数地址的,未来能通过他来调用函数。
  
  那么问题来了,函数是否有地址呢?
  
我们来做个测试:

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

  
在这里插入图片描述
  
  可以看到,我们确实打印出了函数的地址,可见,函数是有地址的
  
  这里与数组有点相似,不论是直接打印数组名还是 &数组名,都能打印出地址,不同的是数组的数组名表示的是数组首元素的地址,而 &数组名 表示的是整个数组的地址,他们仅仅只是在数值上相等,类型是不一样的。而对于函数来说函数名&函数名 的效果是一模一样的。
  
  现在我们把函数的地址取出来了,那该存放在哪呢?老办法,将地址放在指针变量。对于函数的地址,当然是放在函数指针变量中啦,而函数指针变量的写法其实和数组指针变量非常相似(详情请看【C语言】—— 指针三 : 参透数组传参的本质)。
  
如下:

void test()
{printf("hello world\n");
}void (*pf1)() = &test;
void (pf2)() = test;int Add(int x, int y)
{return x + y;
}
int (*pf3)(int x, int y) = Add;
int (*pf4)(int, int) = &Add;//x 和 y 写上或者省略都是可以的

  
  注:函数指针变量中,参数类型的名字可省略,对于函数指针变量来说,重要的是参数类型返回类型,参数名叫什么并不重要。
  
  
函数指针类型解析:

图

  
  学习函数指针后,我们就可以通过函数指针来调用指针指向的函数啦

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

运行结果:
在这里插入图片描述

  
  在这里 Add(1, 2);是通过函数名调用;而(*pf)(2, 3);pf(3, 4);都是通过函数指针调用。
  
  两种函数指针的调用效果是一样的,因此对函数指针来说,解引用可以看作是摆设。
  
  

2.2、两段有趣的代码

  
  接下来,我们来看两段有趣的代码:
  

( *( void( * )( ) ) 0)( );

我们来慢慢分析
  

  • 先来看 void( * )( )部分,是不是觉得很熟呢?,如果你还看不出来,那这样呢: void( *p )( );现在认出来了吧。没错它是一个指针变量,一个变量去掉变量名是什么?是类型。没错 void( * )( )是一个函数指针类型
      
  • 那一个类型加上小括号是什么?是强制类型转换!所以(void(*)())0即是把 0 强制类型转换成函数指针类型(原来是 i n t int int 类型),如果还不理解,我们可以这样来看(void(*)())0x0012ff40。这样是不是清晰了许多呢?
      
  • 既然将 0 强转成函数指针类型,那即意味着 0 地址处放着一个函数,该函数返回类型是 v o i d void void没有参数
      
  • 接着,对位于该地址的函数进行解引用调用该函数:( * ( void( * )( ) ) 0)( ),
      
  • 因为这个函数没有参数(由类型void(*)()可知),所以后面的小括号(参数列表)不填。
      
  • 整句代码的意思是:调用在地址 0 出的函数,该函数的返回类型是 v o i d void void ,没有参数

  

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

这句代码也让我们一起来分析分析
  

  • 先来看里面的部分signal(int,void(*)(int))很明显,signal函数名,而int和void(*)(int)是函数的参数类型
      
  • 那前面的*是什么意思?解引用吗?其实我们不妨顺着函数的思路往下想,一个函数有了函数名参数类型,还差什么?是返回类型。那这个函数的返回类型是什么?剩下的部分就是返回类型
      
  • 其实,该函数的返回类型被劈开了,我们将它的函数名和参数类型拿走,剩下的就是它的返回类型void(*)(int),如果还不清晰,我们可以写成这样来理解(接下来的写法是错误的,仅仅是为了方便理解,题目写法是正确的)void(*)()signal(int,void(*)(int))
      
  • 所以,这一句代码是一个函数声明,函数名是signal;返回类型是void(*)();参数类型是intvoid(*)()

注:以上两段代码均出自 《C陷阱和缺陷》
  
  

三、typedef关键字

3.1、typedef的使用

  
  typedef 是用来类型重命名的,可以将复杂的类型简单化
  
  比如,如果觉得unsigned int写起来不方便,我们可以写成uint就方便多了,那我们可以这样写

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

  
  我们之前简单提到的结构体类型(详情请看【C语言】——详解操作符(下)),觉得每次都要加 s t r u c t struct struct 太麻烦了,那我们可以通过 t y p e d e f typedef typedef 将其重命名。

typedef struct student
{char name[20];int age;
}student;
//将结构体类型struct student重命名为student

  
  那如果是指针类型,可不可以通过 t y p e d e f typedef typedef 来重命名呢?答案是肯定的。比如,将int*重命名成ptr_t,我们可以这样写:

typedef int* ptr_t;

  
  但对于函数指针数组指针稍微有点区别。区别在哪呢?新的类型名的位置不同

  比如我们将数组指针类型int(*)[10]重命名为parr_t,我们可以这么写:

typedef int(*parr_t)[5];

  
  同样,函数指针变量的重命名也是一样的,比如将void(*)(int)重命名为pf_t,可以这样写:

typedef char(*pf_t)(int, int);

  
那么现在,我们就可以用 t y p e d e f typedef typedef 将代码void (*signal(int, void(*)(int)))(int);简化

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

  

3.2、typedef与define比较

  
  想了解 t y p e d e f typedef typedef d e f i n e define define 的区别,我们先来一组比较:

typedef int* ptr_t;
#define PTR_T int*ptr_t p1, p2;
PTR_T p3, p4;

  
他们有什么区别呢?

  • p1p2都是指针变量
  • p3指针变量p4整形变量

为什么会这样呢?
  
  对于ptr_t,他是通过 t y p e d e f typedef typedef 来修饰的, t y p e d e f typedef typedef 的作用就是重命名,因此pyr_t就是int*,他们是画等号的。
  
  而对于PTR_T,他是通过 d e f i n e define define 修饰的,PTR_T仅仅是替换int*int* p3、p4;中, *给了 p3p3指针变量,而p4只剩int了,是整形变量
  
  

四、函数指针数组

  
  数组是一个存放相同类型数据的存储空间,之前,我们已经学过了指针数组(详情请看【C语言】—— 指针二 : 初识指针(下))

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

  
  那要把一个函数的地址放在数组中,这个数组就叫函数指针数组,那函数指针数组该怎么定义呢?

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

答案是: p a r r 1 parr1 parr1
  
  parr1先和[]结合,表示一个parr1是一个数组,那数组中的元素类型是什么呢?是int(*)()类型的函数指针

  
  那么函数指针数组有什么用呢?别急,敬请收看下一章:【C语言】——指针五:转移表与回调函数。
  
  
  
  


  好啦,本期关于字符指针和函数指针就介绍到这里啦,希望本期博客能对你有所帮助,同时,如果有错误的地方请多多指正,让我们在C语言的学习路上一起进步!

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

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

相关文章

解码新时代内存架构:探秘数据在内存中的灵动驻足

欢迎来到白刘的领域 Miracle_86.-CSDN博客 系列专栏 C语言知识 先赞后看&#xff0c;已成习惯 创作不易&#xff0c;多多支持&#xff01; 随着信息技术的飞速发展&#xff0c;我们身处一个数据爆炸的时代。数据的处理和存储方式正日益成为技术革新的重要领域。在新时代的…

CSS时钟案例

文章目录 1. 演示效果2. 分析思路3. 代码实现 1. 演示效果 2. 分析思路 背景是表盘&#xff0c;不用自己制作然后用CSS的定位做时针&#xff0c;分针和秒针黑点用伪元素::after生成转动用animation实现 3. 代码实现 <!DOCTYPE html> <html lang"en">&…

Java学习笔记 | JavaSE基础语法05 | 方法

文章目录 0.前言1. 方法概述2. 方法的定义和调用2.1 无参数方法定义和调用2.2 带参数方法定义和调用1 带参数方法定义和调用2 形参和实参3 带参数方法练习 2.3 带返回值方法的定义和调用1 带返回值方法定义和调用2 带返回值方法练习13 带返回值方法练习24 带返回值方法练习3 3.…

Python学习从0到1 day18 Python可视化基础综合案例 1.折线图

我默记这段路的酸楚&#xff0c;等来年春暖花开之时再赏心阅读 —— 24.3.24 python基础综合案例 数据可视化 — 折线图可视化 一、折线图案例 1.json数据格式 2.pyecharts模块介绍 3.pyecharts快速入门 4.数据处理 5.创建折线图 1.json数据格式 1.什么是json 2.掌握如何使用js…

SqlServer找不到SQL Server Configuration Manager(配置管理)

1、Win键 R &#xff0c;输入 compmgmt.msc 2、找到Sql Server配置管理器

演讲嘉宾公布 | 智能家居与会议系统专题论坛将于3月28日举办

一、智能家居与会议系统专题论坛 智能家居通过集成先进的技术和设备&#xff0c;为人们提供了更安全、舒适、高效、便捷且多彩的生活体验。智能会议系统它通过先进的技术手段&#xff0c;提高了会议效率&#xff0c;降低了沟通成本&#xff0c;提升了参会者的会议体验。对于现代…

Deconstructing Denoising Diffusion Models for Self-Supervised Learning

开头说点题外话&#xff1a;这篇可谓是大咖云集啊&#xff0c;刘壮、谢赛宁、何凯明这些耳熟能详的名字&#xff0c;并且这篇论文一些人也觉得分析特别到位&#xff0c;不愧是大佬视角&#xff0c;配得上“解构”两个字&#xff1b;很巧的是&#xff0c;本科阶段的团队导师也是…

Web框架开发-Ajax

一、 Ajax准备知识:json 1、json(Javascript Obiect Notation,JS对象标记)是一种轻量级的数据交换格式 1 2 它基于 ECMAScript (w3c制定的js规范)的一个子集,采用完全独立于编程语言的文本格式来存储和表示数据。 简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。…

计算机组成原理-5-输入输出系统

5. 输入输出系统 文章目录 5. 输入输出系统5.1 概述5.1.1 I/O系统的发展概况5.1.2 I/O系统的组成5.1.3 I/O设备与主机的连接5.1.4 I/O设备与主机传送信息的控制方式 5.2 I/O设备5.3 I/O接口5.4 控制方式5.4.1 程序查询方式5.4.2 程序中断方式5.4.3 DMA方式5.4.4 程序中断方式与…

RabbitMQ 01

01.定义 02.功能

apifox创建接口含中文字符报错的两种解决方案

针对apifox的含中文报错解决方法&#xff1a; 方法一&#xff1a;创建相应接口后&#xff0c;在设置中URL自动编码为WHATING。 方法二&#xff1a;直接将浏览器的url复制到apifox中&#xff0c;浏览器会自动解析配置中文转换路径。

springboot297毕业生实习与就业管理系统的设计与实现

毕业生实习与就业管理系统 摘 要 使用旧方法对毕业生实习与就业管理系统的信息进行系统化管理已经不再让人们信赖了&#xff0c;把现在的网络信息技术运用在毕业生实习与就业管理系统的管理上面可以解决许多信息管理上面的难题&#xff0c;比如处理数据时间很长&#xff0c;数…

ADS版图优化方法---使用EM-Cosimulation对版图进行OPTIM

ADS版图优化方法—使用EM-Cosimulation对版图进行OPTIM 一般来说&#xff0c;对原理图进行OPTIM优化的方法大伙用的都比较6了&#xff0c;跑起来也非常快。但是得到版图又可能和原理图的结果差的非常大&#xff0c;为了优化版图又不得不重新对原理图的参数进行调谐优化&#x…

这个国产原型设计工具,建议PM新人一定要用!

Hello小伙伴们&#xff01;我是榛妮&#xff0c;原BAT大厂女产品经理一枚&#xff0c;目前在香港创业。 一转眼&#xff0c;做产品经理已经8年&#xff0c;想想入行时的种种往事&#xff08;尴尬情况&#xff09;&#xff0c;至今仍然历历在目。 说起刚入行时遇到的那些问题&a…

蓝桥杯物联网Lora通信功能总结

1、LORA通信在函数LORA被初始化的时候就已经处于接收状态 即开机即能接收数据 2、LORA数据的接收以及发送都通过FIFO数据线 3、LORA的收发同时进行会产生FIFO数据线的通信干扰 4、LORA_Rx在FIFO中有数据的时候才会取出数据&#xff0c;FIFO没有数据会直接跳过 当LORA在发送数…

服务器运行一段时间后

自己记录一下。 一、查看目录占用情况 df -h 命令查看磁盘空间 du -ah --max-depth=1 / 查看根目录下各个文件占用情况 二、mysql日志清空 这个日志是可以清空的 echo > /usr/local/mysql/data/syzl-db2.log #将文件清空 说明: 这个文件这么大是因为,开启 …

TypeScript基础类型

string、number、bolean 直接在变量后面添加即可。 let myName: string Tomfunction sayHello(person: string) {return hello, person } let user Tom let array [1, 2, 3] console.log(sayHello(user))function greet(person: string, date: Date): string {console.lo…

基于python+vue城市交通管理系统的设计与实现flask-django-php-nodejs

此系统设计主要采用的是python语言来进行开发&#xff0c;采用django/flask框架技术&#xff0c;框架分为三层&#xff0c;分别是控制层Controller&#xff0c;业务处理层Service&#xff0c;持久层dao&#xff0c;能够采用多层次管理开发&#xff0c;对于各个模块设计制作有一…

数学建模综合评价模型与决策方法

评价方法主要分为两类&#xff0c;其主要区别在确定权重的方法上 一类是主观赋权法&#xff0c;多次采取综合资讯评分确定权重&#xff0c;如综合指数法&#xff0c;模糊综合评判法&#xff0c;层次评判法&#xff0c;功效系数法等 另一类是客观赋权法&#xff0c;根据各指标…

力扣HOT100 - 15. 三数之和

解题思路&#xff1a; 排序 双指针 注意&#xff1a; 在nums[ k ]&#xff0c;nums[ i ]&#xff0c;nums[ j ]的值与自身重复时均会进行跳过&#xff0c;防止重复添加。 如代码中&#xff1a; 防止nums[ k ]重复&#xff1a;if(k>0&&nums[k]nums[k-1]) continue…