所以我必须读取一个充满这样结构的二进制文件:
typedef struct {
char name[50];
int a,b,c;
int no_args;
} func;
导入的头文件具有如下功能:
void add(int,int);
void min(int,int);
void doSomething();
现在函数名称将始终类似于 add/min/doSomething ... add 的参数为 (a,b),min 的参数为 (b,c)。
那么我如何才能启动这些函数,可能是一个函数的映射? 我想这样做,以便我可以使用相同的大型二进制文件对我的函数进行基准测试。 我将如何在 C 中做到这一点?这是我的程序的简单版本。
func f = {"add",5,8,9,0};
现在我需要启动函数add(a,b);
最佳答案
I need to start the function add(a,b) ;
您可能的意思是“您需要调用名为add
的函数”。
在运行时,C 或 C++ 程序中的函数名称不再重要(并且在概念上不存在)。甚至还有一个 Unix 实用程序,名为 strip ,删除 executable 中的每个名称(以及所有符号表) .
因此,您可以执行以下操作:构造一个将名称(例如 add
等字符串)关联到 function pointers 的数据结构。 。例如,您可以首先为 add
和 sub
的签名定义一个类型:
typedef int sig2t (int, int);
然后,用名称和函数指针填充一个数组;首先声明其类型:
struct funbind_st {
const char*fname;
sig2t* faddr;
};
和数组:
const struct funbind_st funbindings[] = {
{ "add", add },
{ "sub", sub },
{ NULL, (sig2t*)0 }
};
当然,您最好拥有相同(且通用)签名sig2t
的doSomething
。在实践中,您可能想要更高效的东西(可能是一些 hash-table 将名称与函数指针相关联)。
然后,在 funbindings
中查找 fname
的元素 "sub"
字符串是一个简单的练习(您需要 strcmp
比较字符串)。
特别是在 Linux 上,还有另一种方法,即使用 dynamic linker (利用可执行文件的 symbol table ),即 dlopen(3)和 dlsym(3)来自 -ldl
库的函数。
您首先需要使用诸如gcc -rdynamic *.o -ldl -o yourprog
之类的东西链接整个程序
然后您可以使用以下方法获取程序句柄(在 C 代码中):
void* proghdl = dlopen(NULL, RTLD_NOW);
if (!proghdl) {
fprintf(stderr, "dlopen program failed %s\n", dlerror());
exit(EXIT_FAILURE);
}
然后,您可以使用
addptr)名为“add”
的全局函数的地址>
sig2t*addptr = (sig2t*) dlsym(proghdl, "add");
if (!addptr) {
fprintf(stderr, "dlsym add failed %s\n", dlerror());
exit(EXIT_FAILURE);
}
有关详细信息,请阅读文档和 Drepper 的 How to write shared libraries纸。
阅读SICP ,并阅读有关 closures 的更多信息(您需要定义闭包的实现,因为 C 没有任何闭包)和 callbacks 。您可能需要它们。
<小时/>I want to do this so I can benchmark my functions using the same large binary file.
为什么需要是二进制文件。如果您想对具有许多函数的大型库进行基准测试,那么您可能使用了错误的方法。您是否考虑过在程序中嵌入解释器(例如 lua 或 guile )?那么你的基准测试文件将是该解释器中的一些脚本!当然,解释器有其自己的开销,因此您需要确保每个函数都针对足够大的情况进行基准测试(例如,每个函数基准测试需要花费十分之一秒,而不是微秒来运行)。
关于c - 如何使用 Struct 启动函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53132883/