动态函数名称 - 无法解析外部符号

时间:2017-01-05 20:51:42

标签: c gcc linker c-preprocessor unresolved-external

我正在尝试使用预处理程序指令动态生成函数名称(添加可选前缀)。

生成工作,当gcc在预处理器(gcc -E)之后输出代码时,gcc告诉我该函数具有正确的名称。

但是当我在代码中使用这个函数时,gcc会抛出一个带有未解析外部符号的错误? (Link2019 / Link1120)

我不是100%确定链接器是如何工作的,但理论上gcc应该运行预处理器,构建所有的库(更确切地说,首先是原型,然后是主要的定义?)所以应该没有问题? 我必须添加一个特殊的编译器选项吗?一个链接?

我的主要人物:

#define _CRT_SECURE_NO_WARNINGS
#define STRING_FUNCTION_PREFIX my   // Defining a prefix for my string functions
#include <stdlib.h>
#include <stdio.h>
#include <string.h>                 // Original string functions
#include "string.h"                 // My string functions

#define ARRAY_SIZE 50
#define HALLO "HALLO"
#define WELT "WELT"

int main()
{
    char src1[ARRAY_SIZE], src2[ARRAY_SIZE], dst1[ARRAY_SIZE], dst2[ARRAY_SIZE];
    strcpy(src1, HALLO);
    strcpy(dst1, WELT);
    strcpy(src2, HALLO);
    strcpy(dst2, WELT);

    strcat(src1, dst1);
    mystrcat(src2, dst2);

    return 0;
}

我的string.h

#pragma once
#include <stdlib.h>

#if defined STRING_FUNCTION_PREFIX
#define FUNCTION_PASTER(ARG1,ARG2) ARG1 ## ARG2
#define FUNCTION_EVALUATER(ARG1,ARG2) FUNCTION_PASTER(ARG1, ARG2)
#define FUNCTION_NAME(FUNCTION) FUNCTION_EVALUATER(STRING_FUNCTION_PREFIX, FUNCTION)
#else
#define FUNCTION_NAME(FUNCTION) FUNCTION
#endif

/*
 * \brief: Adds the string from src to the destination string
 */
void FUNCTION_NAME(strcat)(char *dst, char *src);

我的string.c

#include "string.h"

void FUNCTION_NAME(strcat)(char *dst, char *src)
{
    int counter = 0, offset = 0;
    while (dst[offset] != '\0')
    {
        offset++;
    }

    dst[offset + counter] = src[counter];
}

使用-E

进行编译时输出string.h
1>  #line 11 "d:\\string.h"
1>
1>
1>
1>
1>  void mystrcat(char *dst, char *src);

感谢您的帮助!

2 个答案:

答案 0 :(得分:2)

STRING_FUNCTION_PREFIX指令在main.c中定义,但不在string.c中定义。因此,当编译string.c时,不会发生替换。如果使用gcc -E编译string.c,您将看到它的效果。

您需要将#define STRING_FUNCTION_PREFIX my放在string.h的顶部而不是main.c.这样,任何需要它的.c文件都定义了它,并且它在所有地方都是一致的。

答案 1 :(得分:1)

您似乎正在尝试在C中创建模板模拟。如果是这种情况,您应该将string.c的内容视为需要包含的头文件由知道STRING_FUNCTION_PREFIX应该是什么价值的人。如果string.c内容是头文件,请将其重命名以使其清除,例如string_template.h

然后,您可以将文件mystring.c实现为:

#define STRING_FUNCTION_PREFIX my
#include "string_template.h"