当前位置:首页 > 科技  > 软件

还是结构体指针写的代码看着舒服

来源: 责编: 时间:2024-05-16 17:42:44 93观看
导读一直觉得C语言较其他语言最伟大的地方就是C语言中的指针,有些人认为指针很简单,而有些人认为指针很难,当然这里的对简单和难并不是等价于对指针的理解程度。为此在这里对C语言中的指针进行全面的总结,从底层的内存分析,彻

一直觉得C语言较其他语言最伟大的地方就是C语言中的指针,有些人认为指针很简单,而有些人认为指针很难,当然这里的对简单和难并不是等价于对指针的理解程度。0Ti28资讯网——每日最新资讯28at.com

为此在这里对C语言中的指针进行全面的总结,从底层的内存分析,彻底让读者明白指针的本质。0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

一、指针变量

0Ti28资讯网——每日最新资讯28at.com

首先读者要明白指针是一个变量,为此作者写了如下代码来验证之:0Ti28资讯网——每日最新资讯28at.com

#include "stdio.h"int main(int argc, char **argv){    unsigned int a = 10;    unsigned int *p = NULL;    p = &a;    printf("&a=%d/n",a);    printf("&a=%d/n",&a);    *p = 20;    printf("a=%d/n",a);    return 0;}

0Ti28资讯网——每日最新资讯28at.com

这说明编译器确实是在解引时无法确定*p的大小,因此这里必须告诉编译器p的类型或者*p的大小,如何告诉呢?很简单,用强制类型转换即可,如下:0Ti28资讯网——每日最新资讯28at.com

*(int*)p

这样上面的程序就可以写为如下:0Ti28资讯网——每日最新资讯28at.com

#include <stdio.h>int main(int argc, char **argv){    int a=10;    void *p;    p=&a;    printf("p=%d/n",*(int*)p);    return 0;}

编译运行后:0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

可以看到结果确实是正确的,也和预期的想法一致。由于void指针没有空间大小属性,因此void指针也没有++操作。0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

六、函数指针

0Ti28资讯网——每日最新资讯28at.com

1. 函数指针使用

函数指针在Linux内核中用的非常多,而且在设计操作系统的时候也会用到,因此这里将详细讲解函数指针。既然函数指针也是指针,那函数指针也占用4个字节(32位编译器)。0Ti28资讯网——每日最新资讯28at.com

下面以一个简单的例子说明:0Ti28资讯网——每日最新资讯28at.com

#include <stdio.h>int  add(int a,int b){    return a+b;}int main(int argc, char **argv){    int (*p)(int,int);    p=add;    printf("add(10,20)=%d/n",(*p)(10,20));    return 0;}

程序运行结果如下:0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

可以看到,函数指针的申明为:0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

函数指针的解引操作与普通的指针有点不一样。0Ti28资讯网——每日最新资讯28at.com

对于普通的指针而言,解引只需要根据类型来取出数据即可,但函数指针是要调用一个函数,其解引不可能是将数据取出,实际上函数指针的解引本质上是执行函数的过程,只是这个执行函数是使用的call指令并不是之前的函数,而是函数指针的值,即函数的地址。0Ti28资讯网——每日最新资讯28at.com

其实执行函数的过程本质上也是利用call指令来调用函数的地址,因此函数指针本质上就是保存函数执行过程的首地址。函数指针的调用如下:0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

为了确认函数指针本质上是传递给call指令一个函数的地址,下面用一个简单例子说明:0Ti28资讯网——每日最新资讯28at.com

0Ti28资讯网——每日最新资讯28at.com

上面是编译后的汇编指令,可以看到,使用函数指针来调用函数时,其汇编指令多了如下:0Ti28资讯网——每日最新资讯28at.com

0x4015e3    mov    DWORD PTR [esp+0xc],0x4015c00x4015eb    mov    eax,DWORD PTR [esp+0xc]0x4015ef    call   eax

分析:第一行mov指令将立即数0x4015c0赋值给寄存器esp+0xc的地址内存中,然后将寄存器esp+0xc地址的值赋值给寄存器eax(累加器),然后调用call指令,此时pc指针将会指向add函数,而0x4015c0正好是函数add的首地址,这样就完成了函数的调用。0Ti28资讯网——每日最新资讯28at.com

细心的读者是否发现一个有趣的现象,上述过程中函数指针的值和参数一样是被放在栈帧中,这样看起来就是一个参数传递的过程。0Ti28资讯网——每日最新资讯28at.com

因此可以看到,函数指针最终还是以参数传递的形式传递给被调用的函数,而这个传递的值正好是函数的首地址。0Ti28资讯网——每日最新资讯28at.com

从上面可以看到函数指针并不是和一般的指针一样可以操作内存,因此作者觉得函数指针可以看作是函数的引用申明。0Ti28资讯网——每日最新资讯28at.com

2. 函数指针应用

在linux驱动面向对象编程思想中用的最多,利用函数指针来实现封装,下面以一个简单的例子说明:0Ti28资讯网——每日最新资讯28at.com

#include <stdio.h>typedef struct TFT_DISPLAY{    int   pix_width;    int   pix_height;    int   color_width;    void (*init)(void);    void (*fill_screen)(int color);    void (*tft_test)(void);}tft_display;static void init(void){    printf("the display is initialed/n");}static void fill_screen(int color){    printf("the display screen set 0x%x/n",color);}tft_display mydisplay={    .pix_width=320,    .pix_height=240,    .color_width=24,    .init=init,    .fill_screen=fill_screen,};int main(int argc, char **argv){    mydisplay.init();    mydisplay.fill_screen(0xfff);    return 0;}

上面的例子将一个tft_display封装成一个对象,上面的结构体成员中最后一个没有初始化,这在Linux中用的非常多。0Ti28资讯网——每日最新资讯28at.com

最常见的是file_operations结构体,该结构体一般来说只需要初始化常见的函数,不需要全部初始化。0Ti28资讯网——每日最新资讯28at.com

上面代码中采用的结构体初始化方式也是在Linux中最常用的一种方式,这种方式的好处在于无需按照结构体的顺序一对一。0Ti28资讯网——每日最新资讯28at.com

3. 回调函数

有时候会遇到这样一种情况,当上层人员将一个功能交给下层程序员完成时,上层程序员和下层程序员同步工作,这个时候该功能函数并未完成,这个时候上层程序员可以定义一个API来交给下层程序员。0Ti28资讯网——每日最新资讯28at.com

而上层程序员只要关心该API就可以了而无需关心具体实现,具体实现交给下层程序员完成即可(这里的上层和下层程序员不指等级关系,而是项目的分工关系)。0Ti28资讯网——每日最新资讯28at.com

这种情况下就会用到回调函数(Callback Function),现在假设程序员A需要一个FFT算法,这个时候程序员A将FFT算法交给程序员B来完成,现在来让实现这个过程:0Ti28资讯网——每日最新资讯28at.com

#include <stdio.h>int  InputData[100]={0};int OutputData[100]={0};void FFT_Function(int *inputData,int *outputData,int num){    while(num--)    {    }}void TaskA_CallBack(void (*fft)(int*,int*,int)){    (*fft)(InputData,OutputData,100);}int main(int argc, char **argv){    TaskA_CallBack(FFT_Function);    return 0;}

上面的代码中TaskA_CallBack是回调函数,该函数的形参为一个函数指针,而FFT_Function是一个被调用函数。0Ti28资讯网——每日最新资讯28at.com

可以看到回调函数中申明的函数指针必须和被调用函数的类型完全相同。0Ti28资讯网——每日最新资讯28at.com

本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-88564-0.html还是结构体指针写的代码看着舒服

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 彻底明白Filter与Interceptor

下一篇: 一个小技巧,写出丝滑的 Python 代码

标签:
  • 热门焦点
  • Raft算法:保障分布式系统共识的稳健之道

    Raft算法:保障分布式系统共识的稳健之道

    1. 什么是Raft算法?Raft 是英文”Reliable、Replicated、Redundant、And Fault-Tolerant”(“可靠、可复制、可冗余、可容错”)的首字母缩写。Raft算法是一种用于在分布式系统
  • Automa-通过连接块来自动化你的浏览器

    Automa-通过连接块来自动化你的浏览器

    1、前言通过浏览器插件可实现自动化脚本的录制与编写,具有代表性的工具就是:Selenium IDE、Katalon Recorder,对于简单的业务来说可快速实现自动化的上手工作。Selenium IDEKat
  • 不容错过的MSBuild技巧,必备用法详解和实践指南

    不容错过的MSBuild技巧,必备用法详解和实践指南

    一、MSBuild简介MSBuild是一种基于XML的构建引擎,用于在.NET Framework和.NET Core应用程序中自动化构建过程。它是Visual Studio的构建引擎,可在命令行或其他构建工具中使用
  • 量化指标是与非:挽救被量化指标扼杀的技术团队

    量化指标是与非:挽救被量化指标扼杀的技术团队

    作者 | 刘新翠整理 | 徐杰承本文整理自快狗打车技术总监刘新翠在WOT2023大会上的主题分享,更多精彩内容及现场PPT,请关注51CTO技术栈公众号,发消息【WOT2023PPT】即可直接领取
  • 让我们一起聊聊文件的操作

    让我们一起聊聊文件的操作

    文件【1】文件是什么?文件是保存数据的地方,是数据源的一种,比如大家经常使用的word文档、txt文件、excel文件、jpg文件...都是文件。文件最主要的作用就是保存数据,它既可以保
  • 为什么你不应该使用Div作为可点击元素

    为什么你不应该使用Div作为可点击元素

    按钮是为任何网络应用程序提供交互性的最常见方式。但我们经常倾向于使用其他HTML元素,如 div span 等作为 clickable 元素。但通过这样做,我们错过了许多内置浏览器的功能。
  • 每天一道面试题-CPU伪共享

    每天一道面试题-CPU伪共享

    前言:了不起:又到了每天一到面试题的时候了!学弟,最近学习的怎么样啊 了不起学弟:最近学习的还不错,每天都在学习,每天都在进步! 了不起:那你最近学习的什么呢? 了不起学弟:最近在学习C
  • 三星Galaxy Z Fold5今日亮相:厚度缩减但仍略显厚重

    三星Galaxy Z Fold5今日亮相:厚度缩减但仍略显厚重

    据官方此前宣布,三星将于7月26日也就是今天在韩国首尔举办Unpacked活动,届时将带来带来包括Galaxy Buds 3、Galaxy Watch 6、Galaxy Tab S9、Galaxy
  • Counterpoint :OPPO双旗舰战略全面落地 高端产品销量增长22%

    Counterpoint :OPPO双旗舰战略全面落地 高端产品销量增长22%

    2023年6月30日,全球行业分析机构Counterpoint Research发布的《中国智能手机高端市场白皮书》显示,中国智能手机品牌正在寻求高质量发展,中国高端智能
Top
Baidu
map