为什么在编译K& R2第1章中的最长行示例时会出现“getline的冲突类型”错误?

时间:2012-01-06 19:07:11

标签: c linux kernighan-and-ritchie

这是一个我试图直接从“C程序设计语言”第1.9节开始运行的程序。

#include <stdio.h>
#define MAXLINE 1000

int getline(char line[], int maxline);
void copy(char to[], char from[]);

main()
{
    int len;
    int max;
    char line[MAXLINE];
    char longest[MAXLINE];

    max = 0;
    while ((len = getline(line, MAXLINE)) > 0)
        if (len > max) {
        max = len;
        copy(longest, line);
        }
    if (max > 0)
        printf("%s", longest);
return 0;
}


int getline(char s[], int lim)
{
    int c, i;

    for (i=0; i<lim-1 && (c=getchar()) !=EOF && c != '\n'; ++i)
        s[i] = c;
    if (c == '\n') {
        s[i] = c;
        ++i;
    }
    s[i] = '\0';
    return i;
}


void copy(char to[], char from[])
{
    int i;

    i = 0;
    while ((to[i] = from[i]) != '\0')
        ++i;
}

以下是我尝试使用Ubuntu 11.10编译程序时遇到的错误:

cc     word.c   -o word
word.c:4:5: error: conflicting types for ‘getline’
/usr/include/stdio.h:671:20: note: previous declaration of ‘getline’ was here
word.c:26:5: error: conflicting types for ‘getline’
/usr/include/stdio.h:671:20: note: previous declaration of ‘getline’ was here
make: *** [word] Error 1

为了确保书中的印刷品没有问题,我从书中引用了章节练习的这组答案(http://users.powernet.co.uk/eton/kandr2 /krx1.html)当我尝试从该链接运行练习18,19,20,21等时,我得到了类似的错误。当我无法运行程序以查看它们的输出时,真的很难学。在一个程序中引入字符数组和函数调用时,就会出现此问题。我对这个问题有任何建议表示感谢。

7 个答案:

答案 0 :(得分:34)

问题是getline()是标准库函数。 (在stdio.h中定义)您的函数具有相同的名称,因此与它发生冲突。

解决方案是简单地更改名称。

答案 1 :(得分:32)

冲突函数getline()是GNU / POSIX扩展名。

K&amp; R声明他们在他们的书(c.f.)中专门针对ANSI C,但没有提供此功能。

  

作者提供了ANSI标准C语言编程的完整指南。

为了将gcc设置为“K&amp; R兼容模式”,您可以指定ANSI或ISO模式进行编译。这些旨在禁用扩展,例如,函数getline()。 这最终可以消除编辑K&amp; R提供的其他示例的需要。

例如,以下编译就好了:

$ gcc test.c -ansi
$ gcc test.c -std=c89

(除非他们抱怨main()-Wall的隐式默认返回类型。)

在某些系统上,这些模式可能无法正常工作(apparantly some version(s) of Mac OS fail to correctly disable all extensions)。我在我的机器上成功测试了这个:

$ gcc --version
gcc (GCC) 4.7.2 20121109 (Red Hat 4.7.2-8)
Copyright (C) 2012 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.

答案 2 :(得分:9)

这是因为stdio.h具有getline()功能。

这样做的一个简单方法就是将您的函数重命名为my_getline()

getline()getdelim()最初都是GNU个扩展名。它们在POSIX.1-2008标准化。

答案 3 :(得分:3)

/usr/include/stdio.h:671:20: note: previous declaration of ‘getline’ was here

那应该给你一个提示。尝试将代码中的getline()函数重命名为其他内容。

此外,以这种方式声明main()是旧式的。默认情况下,没有声明返回类型和参数的函数接受一个未指定数量的参数并返回一个int。这几乎是main()的情况:它确实返回一个int,但有两个参数。你最好把它声明为:

int main(int argc, char **argv)

或:

int main(int argc, char *argv[])

答案 4 :(得分:2)

您必须更改getline的名称,因为它已经存在。

答案 5 :(得分:2)

getline现在是stdio.h

中声明的POSIX函数

getline函数重命名为另一个名称,它将进行编译。

答案 6 :(得分:1)

这已经是一个名为getline的函数,在“stdio.h”文件中定义。因此原型中的冲突! 将您的函数重命名为“my_getline”或其他名称,一切都应该没问题!