1

我正在设计一个图书馆,遇到了一个关于分离公共和图书馆私有操作的问题。我有以下库接口:

libmylib.h

typedef struct lib_context lib_context;

int init_library_context(const char **paths, 
                         size_t path_num, 
                         lib_context **ctx_out);

void release_context(lib_context *ctx);

出于测试目的,我想提供一个为单元测试中使用的不透明结构分配内存的函数。但是由于客户不需要这个功能,我决定将其设为私有(并将结构的定义也放在那里):

lib_context_internal.h

//the definition is simplified for brevity sake
struct lib_context{
   size_t paths_num;
   const char *path[];
};

lib_context * allocate_context(size_t paths_num);

但是函数的定义在libmylib.c(没有lib_context_internal.c文件)中提供:

libmylib.c

#include "limylib.h"
#include "lib_context_internal.h"

lib_context * allocate_context(size_t paths_num){
   size_t size = sizeof(struct lib_context) + sizeof(const char *[paths_num]);
   return calloc(1, size);
}

void release_context(lib_context *ctx){
    //release all strings in the lib_ctx
    free(ctx);
}

int init_library_context(const char **paths, 
                     size_t path_num, 
                     lib_context **ctx_out){
   //open sockets, init epoll, etc...
   *ctx_out = allocate_context(paths_num);
   return 0;
}

困扰我的是我将与相同数据结构相关的分配/释放函数放在不同的头文件中(即使)我在同一个.c文件中定义了它们。做这样的事情不是很常见吗?

4

1 回答 1

4

像这样有一个单独的标题很好。如果您的库中的其他实现文件需要访问其中定义的类型或函数,那么它们也可以包含该标头。只要该内部头文件不与编译库一起分发,就可以了。

我要做的一件事是在私有标头中包含公共标头。这样实现文件只需要担心包含私有头。

lib_context_internal.h:

#include "limylib.h"

//the definition is simplified for brevity sake
struct lib_context{
   size_t paths_num;
   const char *path[];
};

lib_context * allocate_context(size_t paths_num);

libmylib.c:

#include "lib_context_internal.h"

lib_context * allocate_context(size_t paths_num){
   size_t size = sizeof(struct lib_context) + sizeof(const char *[paths_num]);
   return calloc(1, size);
}

...
于 2019-04-01T17:20:27.297 回答