多重定义和第一次在我的 C 程序中定义

Posted

技术标签:

【中文标题】多重定义和第一次在我的 C 程序中定义【英文标题】:Multiple definition and first defined in my C program 【发布时间】:2012-04-05 14:02:59 【问题描述】:

我正在阅读 APUE(Stevens 的“Advanced Programming in the UNIX Environment”),所以有一个文件“apue.h”,其中包含一些声明的自定义函数。我写了一个名为“wait.c”的文件,它定义了在“apue.h”中声明的函数WAIT_CHILDWAIT_PARENT,主函数是14.6.c。

tmp/cciYhMd7.o: In function `err_ret':
14.6.c:(.text+0x0): multiple definition of `err_ret'
/tmp/ccO4WyJS.o:wait.c:(.text+0x0): first defined here

err_ret只是使用了没有定义,有什么问题吗??

等待.c

#include "apue.h"

static volatile sig_atomic_t sigflag;
static sigset_t newmask, oldmask, zeromask;

static void
sig_usr(int signo)

    sigflag = 1;


void
TELL_WAIT(void)

    if(signal(SIGUSR1, sig_usr) == SIG_ERR)
      err_sys("signal(SIGUSR1) error");
    if(signal(SIGUSR2, sig_usr) == SIG_ERR)
      err_sys("signal(SIGUSR2) error");

    sigemptyset(&zeromask);
    sigemptyset(&newmask);

    sigaddset(&newmask, SIGUSR1);
    sigaddset(&newmask, SIGUSR2);

    if(sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0)
      err_sys("SIG_BLOCK error");


void
TELL_PARENT(pid_t pid)

    kill(pid, SIGUSR2);


void
WAIT_PARENT(void)

    while(sigflag == 0)
      sigsuspend(&zeromask);
    sigflag = 0;

    if(sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0)
      err_sys("SIG_SETMASK error");


void
TELL_CHILD(pid_t pid)

    kill(pid, SIGUSR2);


void
WAIT_CHILD(void)

    while(sigflag == 0)
      sigsuspend(&zeromask);
    sigflag = 0;

    if(sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0)
      err_sys("SIG_SETMASK error");

上面的代码是我的“wait.c”源文件。我只使用err_sys。以下是我得到的链接命令行和链接器错误消息:

$ gcc -o a.out wait.c 14.6.c
/tmp/ccZ5F3Pn.o: In function `err_ret':
14.6.c:(.text+0x0): multiple definition of `err_ret'
/tmp/cc5EXGbf.o:wait.c:(.text+0x0): first defined here
/tmp/ccZ5F3Pn.o: In function `err_sys':
14.6.c:(.text+0x38): multiple definition of `err_sys'
/tmp/cc5EXGbf.o:wait.c:(.text+0x38): first defined here
/tmp/ccZ5F3Pn.o: In function `err_exit':
14.6.c:(.text+0x76): multiple definition of `err_exit'
/tmp/cc5EXGbf.o:wait.c:(.text+0x76): first defined here
/tmp/ccZ5F3Pn.o: In function `err_dump':
14.6.c:(.text+0xaf): multiple definition of `err_dump'
/tmp/cc5EXGbf.o:wait.c:(.text+0xaf): first defined here
/tmp/ccZ5F3Pn.o: In function `err_msg':
14.6.c:(.text+0xe6): multiple definition of `err_msg'
/tmp/cc5EXGbf.o:wait.c:(.text+0xe6): first defined here
/tmp/ccZ5F3Pn.o: In function `err_quit':
14.6.c:(.text+0x116): multiple definition of `err_quit'
/tmp/cc5EXGbf.o:wait.c:(.text+0x116): first defined here
/tmp/ccZ5F3Pn.o: In function `TELL_WAIT':
14.6.c:(.text+0x5fe): multiple definition of `TELL_WAIT'
/tmp/cc5EXGbf.o:wait.c:(.text+0x272): first defined here
/tmp/ccZ5F3Pn.o: In function `TELL_CHILD':
14.6.c:(.text+0x72e): multiple definition of `TELL_CHILD'
/tmp/cc5EXGbf.o:wait.c:(.text+0x3a2): first defined here
/tmp/ccZ5F3Pn.o: In function `WAIT_PARENT':
14.6.c:(.text+0x6d8): multiple definition of `WAIT_PARENT'
/tmp/cc5EXGbf.o:wait.c:(.text+0x34c): first defined here
/tmp/ccZ5F3Pn.o: In function `TELL_PARENT':
14.6.c:(.text+0x6bd): multiple definition of `TELL_PARENT'
/tmp/cc5EXGbf.o:wait.c:(.text+0x331): first defined here
/tmp/ccZ5F3Pn.o: In function `WAIT_CHILD':
14.6.c:(.text+0x749): multiple definition of `WAIT_CHILD'
/tmp/cc5EXGbf.o:wait.c:(.text+0x3bd): first defined here
collect2: ld returned 1 exit status

【问题讨论】:

你能告诉我们你的声明和定义部分吗? 问题是函数err_ret存在于其他源文件中,我只是在我的wait.c源文件中使用它。帮帮我! 好的。但是告诉我们你是如何使用它的? 请看我的声明,先谢谢。 向我们展示您的所有代码如何? 【参考方案1】:

不要在头文件中定义函数或变量。在头文件中声明它们,在 .c 源文件中定义它们。

您需要为定义创建一个单独的源文件并单独编译它。

另外,请注意声明全局变量和定义之间的区别。如果你需要声明一个被多个.c文件使用的变量,你必须说:

extern int err__ret;

在头文件中声明变量(作为外部变量),然后添加:

int err_ret;

到一个 .c 文件,定义它。

另请注意,任何定义变量或函数的 .c 文件还应包含声明它的 .h 文件。如果声明和定义之间存在任何差异,这将导致编译器产生错误。

【讨论】:

是的,我的头文件中没有定义函数,所以不知道是什么问题! 在 "apue.h" 中有很多函数声明如err_ret, WAIT_CHILD,其实err_ret是别人写的,我在我的"wait.c"源码中完成WAIT_CHILD文件。【参考方案2】:

APUE 书籍网站的来源在apue.h 中有这三个声明(以及许多其他声明):

void    err_ret(const char *, ...);

void    WAIT_PARENT(void);
void    WAIT_CHILD(void);

从编译器的错误信息来看,不仅定义了err_ret(),实际上还定义了两次,一次在源文件wait.c 中,一次在源文件14.6.c 中。您将需要确定这些定义中的哪一个(如果有的话)是正确的,然后删除另一个,或者决定您不能将wait.c 中的代码与14.6.c 中的程序一起使用。将err_ret() 和朋友放在他们自己的源文件中可能会更好。在源代码中,err_*() 函数在 lib/error.c 中。


我没有重新定义函数err_*(),所以我想我使用lib/error.c 源文件。在apue.h 中有:

#ifndef _APUE_H
#define _APUE_H

所以我认为它不会定义两次。

如果您的 TU(翻译单元)使用 #include "lib/error.c",那么您的 TU 正在定义功能。 C 编译器会看到包含您的源代码的源代码以及您包含的标头以及(假设地)lib/error.c 中的代码。您根本不应该包含该源文件;您应该单独编译它并将目标文件链接到您的程序中。 apue.h 标头声明了 err_*() 函数,因此您可以在代码中使用它们。您需要单独编译 lib/error.c 并将其与您的 wait.o14.6.o 文件链接。

理解定义变量或函数(实际上是为其分配存储空间)和声明变量或函数(告诉编译器该函数或变量存在,但此处未定义)之间的区别至关重要——如在“这个声明没有定义它”中,即使源文件中的下一行确实定义了它)。标头应提供声明,而不应提供定义(大多数情况下——目前这是一条足够好的规则,直到您知道何时以及如何打破规则为止)。


但是当我编写源文件时,我所做的只是#include "apue.h"。我不太明白如何单独编译lib/error.c

所以,听起来您的编译和链接行中需要三个源文件:

gcc -o 14.6 14.6.c wait.c lib/error.c

或者您可以在单独的操作中进行:

gcc -c 14.6.c
gcc -c wait.c
gcc -c lib/error.c -o error.o
gcc -o 14.6 14.6.o wait.o error.o

您可能需要额外的编译器标志(例如-I-D 选项),或额外的库和链接器选项(例如-llib-L /directory/lib 选项)。

如果您的源文件 14.6.c 和 wait.c 包含 #include "apue.h" 并且不包含其他源文件(所以没有 #include "lib/error.c" 或类似的东西 - 并且 14.6.c 不包含您的 wait.c 反之亦然) ,那么您应该不会遇到问题。

但是,您遇到了问题,我们看不到您的源代码或链接命令,这意味着我们必须尝试并猜测您做错了什么。我们可以设计各种奇怪的例子来说明你可能在做什么,但我们很可能是错的。

所以,尽量减少代码,向我们展示您正在编译哪些文件以及如何编译它们。

【讨论】:

其实err_*()的函数我并没有重新定义,所以我想我使用lib/error.c的源文件。而在aoue.h#ifndef _APUE_H #define _APUE_H ,所以我认为它不会定义两次。 如果您的 TU(翻译单元)使用 #include "lib/error.c",那么您的 TU 正在定义功能。 C 编译器会看到包含您的源代码的源代码以及您包含的标头以及(假设地)lib/error.c 中的代码。您根本不应该包含该源文件;您应该单独编译它并将目标文件链接到您的程序中。 apue.h 声明了 err_*() 函数,因此您可以在代码中使用它们。您需要单独编译 lib/error.c 并将其与您的 wait.o14.6.o 文件链接。 但是我做的只是#include "apue.h",当我写我的源文件时,我不太明白如何单独编译lib/error.c 我想再次遵守apue.h 你并没有真正“编译”apue.h;你用它编译。 C 代码文件包括 apue.h 以获取声明,这允许代码使用函数。但是您还必须为函数提供定义,为此,您需要编译包含函数的源文件。【参考方案3】:

请显示 wait.h 的 err_ret() 条目,如果需要,可以隐藏其他行。没有人要求您妥协您的 NDA :)(双关语)。你也有在wait.c 中再次声明err_ret() 吗?如果你有那么它不会编译。(我猜这就是问题所在)

首先进行预处理以找到多个定义,如下所示:

$gcc -E wait.c | grep -n 'err_ret'
$gcc -E 14.6.c | grep -n 'err_ret'

查看输出和行号,找出函数声明和定义的确切位置。

【讨论】:

对不起,我不明白你的意思。帮帮我!【参考方案4】:

我在编译 sigsetjmp.c 时遇到了类似的问题。最后我发现原因是我的apue.h中有#include "error.c",导致编译多个c文件时,每个文件都包含apue.h,然后又包含error.c,导致多重定义err_ret错误.

这是我的建议:

    不要在头文件(apue.h)中包含任何函数定义 c 文件。

    gcc 命令行或 Makefile 中需要的其他 c 文件就可以了。

    例如:gcc sigsetjmp.c pr_mask.c error.c

希望对您有所帮助。

【讨论】:

以上是关于多重定义和第一次在我的 C 程序中定义的主要内容,如果未能解决你的问题,请参考以下文章

变量的多重定义,而它不是

gcc:在 C++ 应用程序中链接 C 库会导致“多重定义”错误

使用 Qt Creator 时出错:“多重定义”“首先在这里定义”

编译C程序出现多重定义,怎么解决?

C ++多重继承,多重定义时没有?

不存在的目录中的多重定义