程序分身2:参考busybox方式实现

2024-02-10 02:32

本文主要是介绍程序分身2:参考busybox方式实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景

约半年前,曾经写过有关程序分身的文章,最近心血来潮翻看busybox的代码,发现原来实现的方式有点笨拙。如busybox名称所示,它将很多的程序都集成到一个程序(box)中,所以非常“busy”。笔者从事多年的嵌入式,发现大部分二进制程序都使用busybox,因为它能大大减少占用空间,即减少flash占用,亦即减少硬件成本。能达到这个目的,一方面利益于其架构,另一方面busybox使用了Linux的链接机制。
本文参照busybox的解析过程,并将其简化。

完整源码

main.c源码如下:

/*** 参考busybox框架代码实现程序分身。* 应用场合:很多业务处理可能需要不同的程序,但又有很多共同部分(如初始化)。* 则可以使用同一套代码,但创建多个链接的方式实现,* 综合考虑了代码组织维护和程序运行两方面。* 具体示例:cgi业务程序* 测试说明:* gcc busybox_main.cpp* cp a.out foo* cp a.out hello* 分别执行foo和hello和a.out(执行a.out没有输出信息)*/
#include <stdio.h>
#include <stdlib.h>
#include <stddef.h>
#include <unistd.h>
#include <string.h>// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
// 以下为不同业务处理模块程序,每个函数可以是独立的,也可以是有部分共用的初始化
// 但不能共享数据,因为对外,它们是单独的程序
int foo(int argc, char *argv[])
{printf("foo\n");return 0;
}int bar(int argc, char *argv[])
{printf("bar\n");return 0;
}int hello(int argc, char *argv[])
{printf("hello\n");for (int i = 0; i < argc; i++){printf("argv[%d]: %s\n", i, argv[i]);}return 0;
}
// >>>>>>>>>>>>>>>>>>>>>>>>>>>> 业务模块函数 结束#define ARRAY_SIZE(x) ((unsigned)(sizeof(x) / sizeof((x)[0])))struct bb_applet
{const char *name; // 程序名称int (* applet_main)(int argc, char* argv[]); // 仿main函数
};// 全局结构体数组,保存程序名称与对应的执行函数
static struct bb_applet g_applets[] =
{{"foo", foo},{"bar", bar},{"hello", hello}
};static int cmp_name(const void *a, const void *b)
{const struct bb_applet *aa = (struct bb_applet *)a;const struct bb_applet *bb = (struct bb_applet *)b;return strcmp(aa->name, bb->name);
}static int applet_name_compare(const void *name, const void *idx)
{int i = (int)(ptrdiff_t)idx - 1;return strcmp((const char*)name, g_applets[i].name);
}int find_applet_by_name(int size, const char *name)
{// 当数量达到一定时,使用二分查找,这样快速,否则直接比较即可if (size > 8){void *p;p = bsearch(name, (void*)(ptrdiff_t)1, size, 1, applet_name_compare);return (int)(ptrdiff_t)p - 1;}else{for (int i = 0; i < size; i++){if (strcmp(name, g_applets[i].name) == 0)return i;}return -1;}
}int main(int argc, char *argv[])
{char* p = NULL;char* applet_name = argv[0];int size = ARRAY_SIZE(g_applets);int applet_no = -1;if ((p = strrchr (applet_name, '/')) != NULL) {applet_name = p + 1;}// 测试代码for (int i = 0; i < size; i++){//printf("org name[%d]: %s\n", i, g_applets[i].name);}// 先进行快排qsort(g_applets, size, sizeof(g_applets[0]), cmp_name);// 测试代码for (int i = 0; i < size; i++){//printf("name[%d]: %s\n", i, g_applets[i].name);}// 再使用二分查找applet_no = find_applet_by_name(size, applet_name);// 如果找到,则执行if (applet_no != -1)g_applets[applet_no].applet_main(argc, argv);return 0;
}

代码不难,首先处理输入程序名称的“./”,因为有时会使用“./a.out”的形式。接着使用快速排序,因为后面可能使用二查找算法查找程序名称。
使用上,按业务需求实现不同的模块函数,将名称与函数放置到bb_applet结构体的全局变量g_applets中。使用创建不同的链接文件(即bb_applet结构体中的name),指向生成的二进制文件。

编译

gcc main.c

执行:

$ ln -s a.out hello // 链接文件
$ ./hello a b c d e 
hello
argv[0]: ./hello
argv[1]: a
argv[2]: b
argv[3]: c
argv[4]: d
argv[5]: e

实践应用、小结

在cgi程序中,web上不同的按钮都对应一个cgi程序,如果分开实现,一些诸如初始化的代码,就要在不同模块中出现。使用本文方法,可减少代码冗余,后期维护也方便。

李迟 2018.1.19 深夜

这篇关于程序分身2:参考busybox方式实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/695931

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

内核启动时减少log的方式

内核引导选项 内核引导选项大体上可以分为两类:一类与设备无关、另一类与设备有关。与设备有关的引导选项多如牛毛,需要你自己阅读内核中的相应驱动程序源码以获取其能够接受的引导选项。比如,如果你想知道可以向 AHA1542 SCSI 驱动程序传递哪些引导选项,那么就查看 drivers/scsi/aha1542.c 文件,一般在前面 100 行注释里就可以找到所接受的引导选项说明。大多数选项是通过"_

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M