【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.rhkb.cn/news/284701.html

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

相关文章

LeetCode Python - 71. 简化路径

目录 题目描述解法运行结果 题目描述 给你一个字符串 path &#xff0c;表示指向某一文件或目录的 Unix 风格 绝对路径 &#xff08;以 ‘/’ 开头&#xff09;&#xff0c;请你将其转化为更加简洁的规范路径。 在 Unix 风格的文件系统中&#xff0c;一个点&#xff08;.&…

【电气安全】ASCP电气防火限流式保护器/末端回路线路保护

为什么要使用电气防火限流式保护器&#xff1f; 应急管理部消防救援局统计&#xff0c;在造成电气火灾事故的原因中&#xff0c;最为主要的当为末端线路短路&#xff0c;在电气火灾事故中占比高达70%以上。如何效预防末端线路短路引发的电气火灾事故&#xff1f; 现阶段最为常…

jmeter断言使用方法

断言主流的有两种&#xff1a;响应断言、JSON断言 响应断言 1、http请求添加响应断言 2、三种作用域&#xff1a;第一种既作用主请求又作用子请求、只作用主请求、只作用子请求。我们默认选中间的仅作用主请求即可。 3、测试字段和匹配规则 测试字段一般选择响应文本即可&am…

Springboot家乡特色推荐系统

目录 背景 技术简介 系统简介 界面预览 背景 在当今这个网络迅猛发展的时代&#xff0c;计算机技术已经广泛应用于我们生活的每个角落&#xff0c;互联网在经济和日常生活等多个方面扮演着至关重要的角色&#xff0c;它已成为人们分享资源和快速交流信息的关键平台。在中国…

灯塔:CSS笔记(5)

定位&#xff1a; 1.标准流 1.块级元素独占一行->垂直布局 2.行内元素/行内块元素一行显示多个->水平布局 2.浮动 可以让原本垂直布局的 块级元素变成水平布局 3.定位 1.可以让元素自由的摆放在网页的任意位置 2.一般用于盒子之间的层叠情况 使用定位的步骤 1.…

基于python+vue超市货品信息管理系统flask-django-php-nodejs

在此基础上&#xff0c;结合现有超市货品信息管理体系的特点&#xff0c;运用新技术&#xff0c;构建了以 python为基础的超市货品信息管理信息化管理体系。首先&#xff0c;以需求为依据&#xff0c;根据需求分析结果进行了系统的设计&#xff0c;并将其划分为管理员和用户二种…

【前端寻宝之路】学习和使用表单标签和表单控件

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法|MySQL| ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-cR8zvB8CkpxTk485 {font-family:"trebuchet ms",verdana,arial,sans-serif;f…

函数作用域和块级作用域:JavaScript中的变量作用域解析

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

tabs自定义样式

使用el-tabs 去修改样式的话比较麻烦&#xff0c;索性直接用div来制作。 <div class"contain"><div class"tab_wrap"><div :class"[skew, first, active 1 ? isActive: ]" click"tabClick(1)"><span class&quo…

Linux(Centos)安装mysql 8 并且使用systemctl管理服务

1.下载mysql包 地址 MySQL :: Download MySQL Community Server (Archived Versions) 注&#xff1a;下载我圈住的减压之后里面会有tar.gz 再次减压才会是软件主体 2.安装和准备 yum -y install numactl 安装numactl tar -xvf mysql-8.0.30-el7-x86_64.tar 拆分 …

C语言操作符和数据类型的存储详解

CSDN成就一亿技术人 目录​​​​​​​ 一.操作符 一.算数操作符&#xff1a; 二.位移操作符&#xff1a; 三.位操作符&#xff1a; 四.赋值操作符&#xff1a; 五.单目操作符&#xff1a; 六.关系操作符&#xff1a; 七.逻辑操作符&#xff1a; 八.条件操作符&…

Docker 笔记(七)--打包软件生成镜像

目录 1. 背景2. 参考3. 文档3.1 使用docker container commit命令构建镜像3.1.1 [Docker官方文档-docker container commit](https://docs.docker.com/reference/cli/docker/container/commit/)Description&#xff08;概述&#xff09;Options&#xff08;选项&#xff09;Exa…

Golang案例开发之gopacket抓包三次握手四次分手(3)

文章目录 前言一、理论知识三次握手四次分手二、代码实践1.模拟客户端和服务器端2.三次握手代码3.四次分手代码验证代码完整代码总结前言 TCP通讯的三次握手和四次分手,有很多文章都在介绍了,当我们了解了gopacket这个工具的时候,我们当然是用代码实践一下,我们的理论。本…

图论基础|695. 岛屿的最大面积、1020. 飞地的数量、130. 被围绕的区域

695. 岛屿的最大面积 力扣题目链接(opens new window) 给你一个大小为 m x n 的二进制矩阵 grid 。 岛屿 是由一些相邻的 1 (代表土地) 构成的组合&#xff0c;这里的「相邻」要求两个 1 必须在 水平或者竖直的四个方向上 相邻。你可以假设 grid 的四个边缘都被 0&#xff0…

何为布控球?布控球的分类对比

主要的分类有&#xff1a; 根据内部的主控板卡的系统分类&#xff0c;典型的是基于海思芯片的嵌入式LINUX系统的&#xff0c;一般出国标GB28181&#xff0c;另外一种是剑走偏锋的安卓系统的&#xff0c;需要把球机的输出YUV转换为UVC接入安卓主板&#xff0c;作为外接USB摄像头…

FreeRTOS从代码层面进行原理分析(1 任务的建立)

FreeRTOS_分析 FreeRTOS 是一个开源的实时操作系统。可以在很的低内存使用的情况下运行在单片机上&#xff0c;使得单片机可以并发(虽然某一时刻还是只有一个任务运行) 的运行程序。关于一些 FreeRTOS 优缺点的介绍文章很多&#xff0c;这里就不再赘述直接深入代码探究原理。 …

vscode用SSH远程开发c语言

vscode配置远程 这里我使用虚拟机进行展示&#xff0c;首先需要你的虚拟机安装好ssh 没安装好就执行下面的命令安装并开启服务 sudo apt-get install ssh sudo service ssh start ps -e | grep sshvscode安装 remote-ssh扩展 点击左下角的远程连接&#xff0c;我这里已经连接…

mysql 存储引擎 基本介绍

目录 一 存储引擎概念介绍 &#xff08;一&#xff09;存储引擎概念 &#xff08;二&#xff09;MySQL常用的存储引擎 &#xff08;三&#xff09;存储引擎运作方式 二 MyISAM 存储引擎介绍 &#xff08;一&#xff09; MyISAM 存储引擎特点 1&#xff0c;不支持…

8.测试教程-自动化测试selenium-3

文章目录 1.unittest框架解析2.批量执行脚本2.1构建测试套件2.2用例的执行顺序2.3忽略用例执行 3.unittest断言4.HTML报告生成5.异常捕捉与错误截图6.数据驱动 大家好&#xff0c;我是晓星航。今天为大家带来的是 自动化测试selenium第三节 相关的讲解&#xff01;&#x1f600…

四、Elasticsearch 进阶

自定义目录 4.1 核心概念4.1.1 索引&#xff08;Index&#xff09;4.1.2 类型&#xff08;Type&#xff09;4.1.3 文档&#xff08;Document&#xff09;4.1.3 字段&#xff08;Field&#xff09;4.1.5 映射&#xff08;Mapping&#xff09;4.1.6 分片&#xff08;Shards&#…