解开 Lua 和 C 代码的纠缠,使 Lua 依赖变为可选项。

我用C编写了一个小程序来管理SQLite3的参考文献数据库。目前这仅仅是一个命令行工具,可以导入和导出BibTeX数据。为了使导出/导入功能更具可定制性(例如,在导入时始终将年份和月份字段组合到日期字段中,或在导出时跳过某些字段),这些功能是用Lua编写的,并从C中调用。现在我想Lua成为可选的依赖项。有什么好的方法? 即,如果在编译时未找到Lua,则回退到更基本的导入/导出功能。

现在,例如,在导出时,我从SQL中获取数据(sqlite3_exec)并在回调函数中将其直接写入Lua表,像这样(剥离堆栈处理;-)):

int db_meta_cb(void *udata, int n, char **cval, char **ckey) {
  while (n-- > 0) {
    lua_pushstring(L, cval[n]);
    lua_setfield(L, -2, ckey[n]);
  };
};

然后有一个Lua函数,它以这样一个表作为参数并漂亮地打印BibTeX条目。导入也类似:BibTeX解析器(Lex/YACC生成的C代码)编写Lua表并调用一个“清除”Lua函数,然后从同一Lua表中读取结果并将其插入数据库。

现在我觉得这种方法不对,因为我可能使用了太多的Lua,我想这是因为数据由(键,值)对组成。

请注意,这个项目的一个原因是尝试/学习嵌入Lua在C中的用法,所以请不要建议(a)使用现有的参考文献管理器之一,或(b)完全重写为Python或Lua ... (我已经做过了)

点赞
用户3323096
用户3323096

我见过的最优雅的方法是通过 动态加载 解决这个问题。不要将程序与 lua 链接,而是在运行时使用 dlopen 函数族手动加载库。

如果尝试加载失败,那么你可以设置一个全局标志,表示你需要采用提供功能的另一种方法,但如果加载成功,则使用 dlsym 函数从 lua 共享对象中获取函数,然后使用它们。你可以创建一个非常优雅的解决方案,其中你可以将一些函数指针分别赋值为 lua 或遗留版本,然后你只需进行一次决策并调用函数,无需关心使用哪种实现。

2014-08-19 10:33:20