出于纯粹的教育目的,我尝试使用“构造函数调用内置函数”编写示例(https://gcc.gnu.org/onlinedocs/gcc-13.2.0/gcc/Constructing-Calls.html#index-_005f_005fbuiltin_005fva_005farg_005fpack)
我写了一个示例,其中包含文档中出现的内容:
#include <stdio.h>
extern int myprintf(FILE *f, const char *format, ...);
extern inline __attribute__((__gnu_inline__)) int myprintf(FILE *f, const char *format, ...)
{
int r = fprintf(f, "myprintf: ");
if (r < 0)
return r;
int s = fprintf(f, format, __builtin_va_arg_pack());
if (s < 0)
return s;
return r + s;
}
int main()
{
myprintf(stdout, "ciao %d\n", 10);
return 0;
}
但我无法建造
max@jarvis:~/test$ gcc --version
gcc (Ubuntu 13.2.0-4ubuntu3) 13.2.0
Copyright (C) 2023 Free Software Foundation, Inc.
This is free software; see the source for copying conditions. There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
max@jarvis:~/test$ gcc -std=gnu2x -Wall test1.c -o test1
/usr/bin/ld: /tmp/ccgD1BYT.o: in function `main':
test1.c:(.text+0x27): undefined reference to `myprintf'
collect2: error: ld returned 1 exit status
gcc 版本如上所示
我哪里错了?
gcc
的默认优化级别是-O0
(无)。如果没有优化,内联不会处于活动状态,因此:发出对 的常规外部调用
myprintf
,该调用没有定义。GCC 手册:3.11 控制优化的选项:
[我的重点]
编译:
或更高的
-O
n,程序将按照您的预期链接并运行。您可以将 assembly 与 assembly 进行比较,
-O0
其中除了外部调用之外没有任何痕迹,而 assembly的myprintf
内联定义-O1
很明显。进一步参考宗派。3.11 如果您想了解在每个优化级别启用了哪些特定的内联选项。
您的函数是
extern inline
,但您没有为您的函数提供外部定义。您的代码无效,您只提供了inline
函数的定义。使用时,
extern inline
您还必须提供该函数的单独的第二个非定义。inline
由于缺少它,编译器可能无法编译undefined reference
该函数的外部定义。只需删除
inline
. 您也可以删除extern
,因为函数是隐式的extern
。只是__attribute__((__gnu_inline__)) int myprintf
,不extern
不inline
。inline
在C中具有非常超级特殊的含义。请勿使用它。您可能有兴趣研究inline
链接,特别是外部内联函数会发生什么?, C中`inline`关键字有什么用?,https://en.cppreference.com/w/c/language/inline。通常情况下,不需要任何特殊的东西。