一些关于Linux内核中的结构体函数指针的理解

部署运行你感兴趣的模型镜像

一些关于Linux内核中常用的结构体函数指针的理解

动机

在看linux内核代码的时候经常能够看到一些结构体里面的成员跟我们以往见的到一些结构体不一样,常见的架构体如下面的代码:

struct a{
    int i;
    char b;
    struct c;
};

而内核中又见有这样的一些结构体:

const struct a{
    .a  = read,
    .b  = write,
    };

这种结构体我很少见,而且很多书籍上没有见过这样的结构体,搜索了一些资料,写下一些对这些结构体的理解

例子

首先放上三段代码作为一个例子
先放上第一段代码:

struct address_space_operations {
    int (*writepage)(struct page *page, struct writeback_control *wbc);
    int (*readpage)(struct file *, struct page *);

    /* Write back some dirty pages from this mapping. */
    int (*writepages)(struct address_space *, struct writeback_control *);

    /* Set a page dirty.  Return true if this dirtied it */
    int (*set_page_dirty)(struct page *page);

    int (*readpages)(struct file *filp, struct address_space *mapping,
            struct list_head *pages, unsigned nr_pages);

    int (*write_begin)(struct file *, struct address_space *mapping,
                loff_t pos, unsigned len, unsigned flags,
                struct page **pagep, void **fsdata);
    int (*write_end)(struct file *, struct address_space *mapping,
                loff_t pos, unsigned len, unsigned copied,
                struct page *page, void *fsdata);

    /* Unfortunately this kludge is needed for FIBMAP. Don't use it */
    sector_t (*bmap)(struct address_space *, sector_t);
    void (*invalidatepage) (struct page *, unsigned int, unsigned int);
    int (*releasepage) (struct page *, gfp_t);
}

这是第二段代码:

const struct address_space_operations v9fs_addr_operations = {
    .readpage = v9fs_vfs_readpage,
    .readpages = v9fs_vfs_readpages,
    .set_page_dirty = __set_page_dirty_nobuffers,
    .writepage = v9fs_vfs_writepage,
    .write_begin = v9fs_write_begin,
    .write_end = v9fs_write_end,
    .releasepage = v9fs_release_page,
    .invalidatepage = v9fs_invalidate_page,
    .launder_page = v9fs_launder_page,
    .direct_IO = v9fs_direct_IO,
};

这是第三段代码:

static int v9fs_vfs_readpage(struct file *filp, struct page *page)
{
    return v9fs_fid_readpage(filp->private_data, page);
}

首先,我们可以看到,我们常见的那种结构体形式就是第二段代码中的这种形式。而弄懂这种形式就是我们这个文章的目的。
我们先需要知道的是,这个结构体里面的东西是函数指针,也就是说,这个成员的变量是跟函数有关的。

我们先看第二段代码,第二段代码除了里面的东西以外,外面的那层还是我们熟悉的东西:结构体

const struct address_space_operations v9fs_addr_operations

看到这个,我们或许知道了v9fs_addr_operations是类型为结构体address_space_operations的变量。

好了,现在我们知道v9fs_addr_operations的类型是结构体address_space_operations,那我们就看看第一段代码:
第一段代码里面,我们看到了很多声明,例如

int (*readpage)(struct file *, struct page *);

这里就是声明了结构体中address_space_operations的一个readpage的函数指针,说明了结构体address_space_operations中有一个名为readpage的,入参为(struct file *, struct page *)的一个函数。

至此,我们就知道了这个结构体里面的声明是咋回事了。

然后我们再看回第二段代码,这时我们就知道了v9fs_addr_operations中的readpage是指向函数v9fs_vfs_readpage的了。但是v9fs_vfs_readpage又是什么东西呢?这个时候请看第三段代码。
这里就实现了v9fs_vfs_readpage。v9fs_vfs_readpage里面实际调用的是v9fs_fid_readpage(filp->private_data, page);这个函数。
因此,大体的流程我归纳如下:
v9fs_addr_operations.readpage(filp->private_data, page) -> v9fs_fid_readpage(filp->private_data, page);

那为什么要搞的这么复杂呢?
我个人认为,这种方法有点类似于面向对象的抽象。把一大类具有相同特征的操作抽象出来,用我们文章中的例子,对于地址空间的操作,基本就是读、写、设脏等等的操作,这样可以把这些操作抽象成一个对象,在这里用一个结构体struct address_space_operations来搞定,这样,address_space_operations就是这些操作的一个父类。

然后现在要对v9fs进行操作,我们只需要实例化一个具体针对v9fs的对象,这时候我们就生成一个类型为struct address_space_operations的对象v9fs_addr_operations。在里面将需要调用的操作指向我们具体的函数,就能实现对于多种函数实现的情景下统一接口拉

您可能感兴趣的与本文相关的镜像

ACE-Step

ACE-Step

音乐合成
ACE-Step

ACE-Step是由中国团队阶跃星辰(StepFun)与ACE Studio联手打造的开源音乐生成模型。 它拥有3.5B参数量,支持快速高质量生成、强可控性和易于拓展的特点。 最厉害的是,它可以生成多种语言的歌曲,包括但不限于中文、英文、日文等19种语言

<think>好的,用户现在想了解如何在C语言中初始化和使用结构体中的函数指针。之前我已经给出了一些方法,但需要结合用户提供的引用内容来调整回答,确保准确性和相关性。 首先,用户提供的引用[1]提到了结构体指针和成员指针的初始化,比如使用malloc分配内存,并强调了成员指针同样需要初始化。这说明用户可能关心内存管理和初始化步骤的细节,需要在我的回答中明确这部分内容,尤其是动态分配内存时的注意事项。 引用[2]讨论了函数指针结构体中的应用,特别是在Linux内核驱动开发中的重要性,提到通过给结构体成员赋值函数地址来实现功能。这提示用户可能对实际应用场景,尤其是嵌入式或系统编程感兴趣,因此需要结合实例说明如何将函数指针赋值给结构体成员,并展示实际调用过程。 引用[3]涉及结构体指针的初始化和数组处理,比如结构体数组的指针指向首地址。这可能意味着用户对结构体指针的基础操作需要进一步澄清,尤其是在静态初始化和动态分配时的不同方法。需要比较这两种情况,说明如何正确初始化结构体指针,并避免常见错误。 用户的上一次问题中,我详细介绍了三种方法:直接赋值、大括号初始化和动态内存分配。现在需要将这些方法整合,并参考用户提供的引用内容,补充内存管理部分,比如malloc和calloc的使用,以及释放资源的必要性。同时,需要强调结构体成员指针的初始化,如引用[1]中提到的stu1->name的初始化,这可能涉及到嵌套指针的情况,需要特别说明。 用户可能存在的深层需求是确保结构体中的函数指针正确指向有效的函数,避免空指针或未定义行为。因此,在回答中需要加入错误检查,例如在动态分配内存后检查指针是否为NULL,并演示如何安全地使用这些函数指针。 此外,用户可能希望了解如何在实际项目中组织代码,比如将函数定义和结构体声明分开,使用头文件等,以提高代码的可维护性。可以简要提到模块化设计,但主要焦点还是初始化和使用步骤。 最后,根据用户之前的回答,相关问题需要围绕初始化错误、性能优化和应用场景展开,结合引用中的内核驱动开发,提出相关问题,帮助用户进一步探索该主题。 总结回答结构:先介绍基本概念,然后分静态初始化和动态初始化两种情况,结合代码示例,强调内存管理和错误处理,最后总结注意事项和常见问题。</think>### 结构体函数指针的初始化与使用方法(C语言) 在C语言中,结构体内嵌函数指针的初始化需要明确**函数指针类型匹配**和**内存管理**两大核心问题。以下是具体实现方法及注意事项: --- #### 一、基础概念 1. **函数指针定义** 结构体中的函数指针需与目标函数**签名严格一致**(参数类型、返回值类型)。 例如:定义接收`int`参数且无返回值的函数指针: ```c typedef void (*FuncPtr)(int); // 类型别名 typedef struct { FuncPtr action; // 结构体中的函数指针成员 int data; } DeviceDriver; ``` 2. **内存分配原则** - **静态初始化**:全局/静态结构体变量,编译器自动分配内存 - **动态初始化**:需手动通过`malloc`或`calloc`分配内存,并手动释放(`free`) --- #### 二、初始化方法 ##### 方法1:静态初始化(编译期) 直接通过大括号`{}`初始化函数指针,适用于全局或局部静态变量。 ```c #include <stdio.h> // 定义函数 void led_on(int brightness) { printf("LED亮度设置为:%d\n", brightness); } // 定义结构体类型 typedef struct { void (*control)(int); int param; } HardwareController; int main() { // 静态初始化结构体 HardwareController ctrl = {led_on, 100}; // 函数指针直接赋值 ctrl.control(ctrl.param); // 输出:LED亮度设置为:100 return 0; } ``` **特点**:无需内存管理,但灵活性较低[^3]。 --- ##### 方法2:动态初始化(运行时) 通过动态内存分配实现,需手动管理内存生命周期。 ```c #include <stdio.h> #include <stdlib.h> // 函数定义 void motor_start(int speed) { printf("电机启动,转速:%d RPM\n", speed); } typedef struct { void (*start)(int); int config; } MotorDriver; int main() { // 动态分配结构体内存 MotorDriver *driver = (MotorDriver*)malloc(sizeof(MotorDriver)); if (!driver) { perror("内存分配失败"); return -1; } // 初始化成员 driver->start = motor_start; // 绑定函数指针 driver->config = 2000; // 调用函数指针 driver->start(driver->config); // 输出:电机启动,转速:2000 RPM // 释放内存 free(driver); return 0; } ``` **关键点**: - 必须检查`malloc`返回值,避免空指针解引用[^1] - 若结构体包含**嵌套指针成员**(如`char* name`),需逐层初始化(见下文) --- ##### 方法3:复合字面量(C99+) 适用于一次性初始化结构体,无需预先分配变量名。 ```c typedef struct { int (*calculate)(int, int); } MathOperation; int add(int a, int b) { return a + b; } int main() { MathOperation op = { .calculate = add }; // 复合字面量初始化 printf("结果:%d\n", op.calculate(3,5)); // 输出:8 return 0; } ``` --- #### 三、嵌套指针的初始化 若结构体包含指针成员(如字符串),需**逐层分配内存**: ```c typedef struct { char* name; // 嵌套指针 void (*log)(char*); // 函数指针 } Logger; void write_log(char* msg) { printf("[LOG] %s\n", msg); } int main() { Logger *logger = (Logger*)calloc(1, sizeof(Logger)); logger->name = (char*)malloc(32 * sizeof(char)); strcpy(logger->name, "SystemLogger"); logger->log = write_log; logger->log(logger->name); // 输出:[LOG] SystemLogger // 释放顺序:先内层,后外层 free(logger->name); free(logger); return 0; } ``` **陷阱**: - 未初始化`name`直接使用会导致段错误 - 内存泄漏(忘记释放嵌套指针) --- #### 四、内核驱动开发中的典型应用 Linux内核中常用结构体函数指针实现**设备驱动接口**,例如: ```c // 模拟字符设备驱动 struct file_operations { ssize_t (*read)(struct file*, char*, size_t, loff_t*); ssize_t (*write)(struct file*, const char*, size_t, loff_t*); int (*open)(struct inode*, struct file*); }; // 具体驱动实现 static ssize_t my_read(struct file* fp, char* buf, size_t len, loff_t* offset) { /* 读取硬件数据到buf */ return len; } // 初始化结构体 struct file_operations fops = { .read = my_read, .write = NULL, // 未实现写操作 .open = generic_open }; ``` **设计模式**:通过填充函数指针实现**多态**,同一接口支持不同硬件[^2]。 --- #### 五、常见问题与调试 1. **段错误(Segmentation Fault)** - 原因:未初始化函数指针或内存访问越界 - 调试:使用`gdb`检查指针是否为`NULL` 2. **函数指针类型不匹配** - 错误示例:将`void(*)(int)`赋值给`int(*)(int)` - 编译器警告:`incompatible pointer types` 3. **内存泄漏检测工具** - Valgrind:检查未释放的动态内存 - 示例命令:`valgrind --leak-check=full ./your_program` --- ### 总结表格 | **场景** | **初始化方法** | **内存管理** | 典型应用 | |------------------|------------------------------|---------------------|-----------------| | 全局/静态结构体 | 大括号`{}`直接赋值 | 自动 | 配置参数存储 | | 动态创建结构体 | `malloc` + 手动赋值 | 需`free` | 设备驱动实例 | | 内核驱动开发 | 复合字面量或静态初始化 | 通常由内核管理 | `file_operations` | --- ### 相关问题 1. 如何避免结构体函数指针的“野指针”问题? 2. 在动态链接库(DLL)中如何使用结构体函数指针实现插件架构? 3. 函数指针与回调机制在事件驱动编程中有哪些具体应用?
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值