c - 如何使用 Struct 启动函数?

标签 c

所以我必须读取一个充满这样结构的二进制文件:

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 的数据结构。 。例如,您可以首先为 addsub 的签名定义一个类型:

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 }
};

当然,您最好拥有相同(且通用)签名sig2tdoSomething。在实践中,您可能想要更高效的东西(可能是一些 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.

为什么需要是二进制文件。如果您想对具有许多函数的大型库进行基准测试,那么您可能使用了错误的方法。您是否考虑过在程序中嵌入解释器(例如 luaguile )?那么你的基准测试文件将是该解释器中的一些脚本!当然,解释器有其自己的开销,因此您需要确保每个函数都针对足够大的情况进行基准测试(例如,每个函数基准测试需要花费十分之一秒,而不是微秒来运行)。

关于c - 如何使用 Struct 启动函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53132883/

相关文章:

c - 如何给空数组赋值

字符与数字的比较

c - 在 C 中创建套接字时出错 (Visual Studio 2017)

c - 如何实现与push相反的功能?

c - 环绕数组导致c中的意外输出

c - 在 Linux 中接收 SIGINT 和异常句柄

c - 如何从 C 中作为结构成员的指针访问数组中的值

c - 关于嵌入式 C 中的定时器

c - sizeof(string) 不包含 "\"符号

c - fgets 如何从文件中逐行读取?