本文地址:http://www.cnblogs.com/yhLinux/p/4079930.html
问题描述:
在练习《UNIX环境高级编程》APUE程序清单8-7的时候,codelist8-7.c中用到了codelist15-3.c中的函数TELL_WAIT(),WAIT_PARENT()及TELL_CHILD()。
codelist8-7.c:
#include "apue.h" static void charatatime(char *); int main(void)
{
pid_t pid;
TELL_WAIT(); if ((pid = fork()) < ) {
err_sys("fork error");
} else if (pid == ) {
WAIT_PARENT(); /* parent goes first */
charatatime("output from child\n");
} else {
charatatime("output from parent\n");
TELL_CHILD(pid);
}
exit();
} static void charatatime(char *str)
{
char *ptr;
int c; setbuf(stdout, NULL); /* set unbuffered */
for (ptr = str; (c = *ptr++) != ; )
putc(c, stdout);
}
codelist8-7.c
codelist15-3.c:
#include "apue.h" static int pfd1[], pfd2[]; void TELL_WAIT(void)
{
if (pipe(pfd1) < || pipe(pfd2) < )
err_sys("pipe error");
} void TELL_PARENT(pid_t pid)
{
if (write(pfd2[], "c", ) != )
err_sys("write error");
} void WAIT_PARENT(void)
{
char c; if (read(pfd1[], &c, ) != )
err_sys("read error"); if (c != 'p')
err_quit("WAIT_PARENT: incorrect data");
} void TELL_CHILD(pid_t pid)
{
if (write(pfd1[], "p", ) != )
err_sys("write error");
} void WAIT_CHILD(void)
{
char c; if (read(pfd2[], &c, ) != )
err_sys("read error"); if (c != 'c')
err_quit("WAIT_CHILD: incorrect data");
}
在使用命令编译8-7时,提示以下错误:
$ gcc codelist8-.c codelist15-.c -o -
/tmp/ccMDAwpv.o: In function `err_ret':
codelist15-.c:(.text+0x0): multiple definition of `err_ret'
/tmp/ccXi2EPL.o:codelist8-.c:(.text+0x0): first defined here
/tmp/ccMDAwpv.o: In function `err_sys':
codelist15-.c:(.text+0xa9): multiple definition of `err_sys'
/tmp/ccXi2EPL.o:codelist8-.c:(.text+0xa9): first defined here
/tmp/ccMDAwpv.o: In function `err_exit':
codelist15-.c:(.text+0x15a): multiple definition of `err_exit'
/tmp/ccXi2EPL.o:codelist8-.c:(.text+0x15a): first defined here
/tmp/ccMDAwpv.o: In function `err_dump':
codelist15-.c:(.text+0x209): multiple definition of `err_dump'
/tmp/ccXi2EPL.o:codelist8-.c:(.text+0x209): first defined here
/tmp/ccMDAwpv.o: In function `err_msg':
codelist15-.c:(.text+0x2b5): multiple definition of `err_msg'
/tmp/ccXi2EPL.o:codelist8-.c:(.text+0x2b5): first defined here
/tmp/ccMDAwpv.o: In function `err_quit':
codelist15-.c:(.text+0x360): multiple definition of `err_quit'
/tmp/ccXi2EPL.o:codelist8-.c:(.text+0x360): first defined here
collect2: ld 返回
查找网上意见如下:
1. http://bbs.chinaunix.net/thread-3699788-1-1.html
我想是不是因为我在apue.h头文件中,添加了#include "error.c",虽然apue.h中 #ifndef __APUE_H__
#define __APUE_H__ 复制代码
但是编译器对每个文件是分别编译的,所以在文件wait.c和14..c中都#include "apue.h",就会包含两份error.c文件,
而在error.c文件中是函数的定义(并不是声明),所以才会出现这样的情况。 所以我删除在apue.h中#include "error.c",makefile文件如下: inc=/home/lee/program/apue/apue.2e/include/
error=/home/lee/program/apue/apue.2e/include/error.c
a.out:14.6.c wait.c
gcc -I $(inc) -o a.out 14.6.c wait.c $(error) 复制代码
apue.h文件中/home/lee/program/apue/apue.2d/include/目录下。
这样就没有问题了。
不知是不是如我想的这样。 #没错,而且没有充分理由时尽量不要 include c 文件
以上这条讨论讲得比较到位吧,原来,我之前按这篇文章的方法[http://blog.csdn.net/quan9ing007/article/details/9929659此方法不好]把 apue.h 和 error.h 都拷贝到 /usr/include 文件夹下了。
其实按上面的说法,不该把在apue.h中#include "error.c",并把 error.c 放到 /usr/include 目录下的,在每一次编译时添加error.c就好。
解决方案(推荐):
因此,只把 apue.h 放到/usr/include目录下,而由于要经常用到error.c,我们将定义一个error环境变量,这样就不必每次都把error.c拷贝到相关文件夹下参与编译。
这里假定当前用户是Lee,error.c存放在/home/Lee/code_Lee/APUE/part_of_source/:
sudo cp /home/Lee/code_Lee/APUE/part_of_source/apue.h /usr/include/apue.h sudo chmod a+r /usr/include/apue.h vi /home/Lee/.bashrc 在.bashrc文末添加apue_error变量: apue_error=/home/Lee/code_Lee/APUE/part_of_source/error.c source ~/.bashrc /* 这很重要,一定要执行 */ echo ${apue_error} /home/Lee/code_Lee/APUE/part_of_source/error.c gcc codelist8-.c ${apue_error} -o - 成功!
gcc codelist8-.c codelist15-.c ${apue_error} -o - 成功!!
(完)
参考资料:
1. Linux的环境变量
http://www.cnblogs.com/Neddy/archive/2011/03/01/1968018.html
2. linux环境变量(转)
http://www.cnblogs.com/growup/archive/2011/07/02/2096142.html