我正在尝试使用预处理程序指令动态生成函数名称(添加可选前缀)。
生成工作,当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.h1> #line 11 "d:\\string.h"
1>
1>
1>
1>
1> void mystrcat(char *dst, char *src);
感谢您的帮助!
答案 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"