- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
所以我必须读取一个充满这样结构的二进制文件:
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/
我有一个数组 items[] items[] 中的每一项都是一个结构体。 item 有键 id、date、value(即 item.id、item.date、item.value) 我想使用 Stru
我想存储 100 名员工。 RollNo,姓名,工资,时间(各种数据,我无法在这里解释,但你可以看下面的代码片段来理解 main() { struct day { int hour
这个问题在这里已经有了答案: storage size of ‘names’ isn’t known (3 个答案) 关闭 5 年前。 我正在尝试蓝牙编程,遇到了这个我不明白的问题。基本上,当我使用
这是一个奇怪的事情: 我有一个结构,它包含指向相同类型结构的指针和指向其他类型结构的指针,以及一些其他值。 struct animal { struct animal * father;
我有一个结构定义如下(名称不同) struct str1 { int field1; struct str2; } 我在一个函数中有一个*str1。我想要一个指向 str2 的指针。 所以
DISK_DETECTION_INFO is defined as有什么原因吗? typedef struct _DISK_DETECTION_INFO { DWORD Size
我正在尝试打包一个字符串和一个字符串的长度。 fmt = '
我在创建结构时遇到问题。 我的结构: public struct Device: Codable { let data: DeviceData let meta: Meta? } pu
struct Item { var name:String? var type:String? var value:Int? var tag:Int? } ... ..
// NewReaderSize returns a new Reader whose buffer has at least the specified 43 // size. If the ar
这个问题在这里已经有了答案: Sorting a vector of custom objects (14 个答案) 关闭 3 年前。 在下面的 C++ 片段中, 如何基于 TwoInts 结构中的
#include struct Header { unsigned long long int alignment; }; int main(void) { struct Heade
我有一个目前看起来像这样的结构(缩写为仅显示基本部分): typedef struct { uint32_t baudrate; ... some other internally u
对此没有太多解释,这就是我所拥有的: public struct PACKET_HEADER { public string computerIp; publi
我有以下代码: struct MyStruct{ data: &'a str, } fn get(S: &'a MyStruct) -> &'a str{ S.data } fn se
struct S1 { char c; int i; }; struct S3 { char c1; struct S1 s; double c2; }; 我正
我有一个名为 Parameter 的协议(protocol): protocol Parameter { var name: String { get } var unit: Unit
有 2 个 struct 定义 A 和 A。我知道 struct A 可以包含指向 struct A 的 POINTER 但我不明白为什么 struct A 不能包含struct A(不是指针) 最佳
我有以下代码: struct MyStruct{ data: &'a str, } fn get(S: &'a MyStruct) -> &'a str{ S.data } fn se
为了说明这一点,这里有一个小的不可变结构和一个更新它的函数: (struct timeseries (variable observations) #:transparent) (define (ad
我是一名优秀的程序员,十分优秀!