news 2026/3/17 17:52:55

从 Android 回调到 C 接口:函数指针 + void* self 的一次彻底理解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
从 Android 回调到 C 接口:函数指针 + void* self 的一次彻底理解

一、为什么学 C 回调会觉得“抽象”?

很多 Android / Java 开发者在第一次接触 C 回调时,都会被下面这种代码劝退:

typedef struct { void (*open)(void* self); void (*close)(void* self); } Ops;

看起来既不像函数,也不像类,更不像接口。

但当我真正理解它之后,才发现:

这套写法,本质上和 Android 的接口回调是完全一致的,只是语法层面更底层。

二、从 Android 的 setOnClickListener 说起

在 Android 中,我们非常熟悉这种写法:

button.setOnClickListener(listener);

这里发生了三件事:

  1. listener是一个对象实例

  2. 它实现了OnClickListener接口

  3. 系统在合适的时机回调:

listener.onClick(view);

关键点在于:

  • 系统并不关心 listener 的具体类型

  • 只关心:你有没有实现规定的方法

三、C 语言里没有接口,那怎么办?

C 语言没有:

  • class
  • interface
  • this

但系统级代码同样需要:

  • 回调
  • 解耦
  • 多态

于是 C 选择了一种“手工实现接口”的方式:

struct + 函数指针

四、Ops:C 里的“接口定义”

typedef struct { void (*open)(void* self); void (*close)(void* self); } Ops;

这段代码不是在实现逻辑,而是在做一件事:

定义一组能力约定(接口)

可以直接在脑子里翻译成 Java:

interface Ops { void open(); void close(); }

区别只有一个:

  • Java 有隐式this
  • C 需要显式传self

五、void* self 是什么?

void* self的本质是:

C 版的 this / Context / 上下文对象

因为 C 没有对象模型,所以:

  • 谁在调用

  • 状态在哪里

都必须由调用方手动传入

void file_open(void* self) { File* f = (File*)self; printf("%d\n", f->fd); }

这和 Java 里的:

void open() { System.out.println(this.fd); }

在“角色”上是完全等价的。

六、接口是怎么“绑定实现”的?

很多人会问:

file_open为什么会成为open的实现?

答案不在函数名,而在赋值

Ops file_ops = { .open = file_open, .close = file_close };

这一刻发生了绑定:

  • file_ops.open指向file_open

  • file_ops.close指向file_close

之后的调用:

file_ops.open(&f);

等价于:

file_open(&f);

七、file_ops 是什么角色?

可以非常准确地说:

file_ops是一个“接口实例 / 方法表 / 回调对象”

它就像 Android 里的:

OnClickListener listener = new MyClickListener();

只不过在 C 里:

  • 方法表(Ops)

  • 对象实例(self)

被拆开保存。

八、一句话总结构成完整模型

**C 的接口回调 = struct(接口定义)

  • 函数指针(方法)

  • void* self(this / Context)**

这并不是“奇怪的 C 写法”,而是:

在没有语言级支持的情况下,手工实现的面向对象与接口机制。

九、写在最后

当我把 C 的这套回调模型和 Android 的接口机制对齐之后,才真正意识到:

语言不同,但工程思想是完全相通的。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/13 22:10:01

网站健康度核心:失效链接的系统性诊断与修复完整方案

失效链接是指您网站上的超链接,导致网页不存在或无法访问,点击时常常返回404错误。 失效链接会让无法访问所需内容的访客感到沮丧,也会损害你的SEO,因为搜索引擎无法有效爬取页面并传递链接权益。 下面我们将更详细地解释什么是失…

作者头像 李华
网站建设 2026/3/13 16:37:33

【护理Agent效率革命】:基于20年临床经验的7种提醒频率场景适配方案

第一章:护理Agent提醒频率的核心价值与挑战在智能护理系统中,护理Agent的提醒频率直接影响用户体验与健康干预效果。过高频率可能导致信息疲劳与用户抵触,而过低则可能错过关键护理时机。因此,设定合理的提醒策略是实现高效人机协…

作者头像 李华
网站建设 2026/3/13 23:04:59

Reactor Core 性能测试完全指南:JMH实战与优化策略

Reactor Core 性能测试完全指南:JMH实战与优化策略 【免费下载链接】reactor-core Non-Blocking Reactive Foundation for the JVM 项目地址: https://gitcode.com/gh_mirrors/re/reactor-core 在现代高并发应用开发中,性能优化是每个开发者必须面…

作者头像 李华
网站建设 2026/3/17 16:30:20

掌握gsplat相机模型:从入门到实战的完整配置指南

掌握gsplat相机模型:从入门到实战的完整配置指南 【免费下载链接】gsplat CUDA accelerated rasterization of gaussian splatting 项目地址: https://gitcode.com/GitHub_Trending/gs/gsplat 在3D渲染技术快速发展的今天,gsplat作为CUDA加速的高…

作者头像 李华
网站建设 2026/3/13 18:45:26

手机拍照一点个人观点

现在手机拍照已经向吃饭一样走入每个的日常生活,现在手机拍照都想拍得漂亮得体,这就考验每个人的拍摄水平,普通人没有拍照技巧,这就是问题所在,怎样让普通人拍出好的照片,关于这点我谈谈个人的一点思路&…

作者头像 李华
网站建设 2026/3/16 1:39:02

地平线生存之道

在辅助驾驶的竞技场上,残酷性不仅体现在技术迭代的速度,更在于商业模式的快速重构。当车企每个季度投入数十亿元只为让智驾系统更拟人,芯片算力需求以指数级膨胀,L2级尚在普及中挣扎,行业已悄然将目光投向L3、L4的更高…

作者头像 李华