C程序printf的奇怪行为

时间:2013-03-26 13:16:07

标签: c printf

我正在执行UNIX wc命令的简单实现,我有一个非常奇怪的问题。如果我删除第49行上的printf,程序将无法正常工作。我尝试了很多东西,但没有一个能奏效。该程序有效,但输出不是我需要和想要的。

#include<stdio.h>
#include<stdlib.h>
#include<fcntl.h>
#include <string.h>
#include<errno.h>
#include<sys/stat.h>

int *counter(FILE* file) {

    // Function counter - counts the number of lines, words and symbols
    int nsymbols = 0, nwords = 0, nlines = 0;
    char c;
    while (EOF != (c = fgetc(file))) {
        nsymbols++;
        if(c == ' '){
            nwords++;
        }
        if(c == '\n'){
            nlines++;
            nwords++;
        }
    }
    int count[] = {nlines, nwords, nsymbols};
    return count;
}

int main(int argc,char* argv[]) {
    if(argc == 1) {
        int *counted;
        counted = counter(stdin);
        printf("\t%d \t%d \t%d\n", counted[0], counted[1], counted[2]);
    }
    else{
        int i,k, bool = 0;
        int total[] = {0,0,0};
        char c = ' ', w = ' ', l = ' ';

        for(i = 1; i < argc; i++) {

            // Cheking if there are some options activated
            if(strchr(argv[i], '-')){
                if(strcmp(argv[i], "-")==0){
                    continue;
                }
                if(strchr(argv[i], 'l')) {
                    l = 'l';
                }
                if(strchr(argv[i], 'w')) {
                    w = 'w';
                }
                if(strchr(argv[i], 'c')){
                    c = 'c';
                }
                bool = 1;
            }
        }
        if(!bool) {
            // If none of them are activated from the user, automatically activates all of them
            l = 'l';
            w = 'w';
            c = 'c';
        }
        printf("%c %c %c", l,w,c);

        for(i = 1; i<argc; i++){

            if(strcmp(argv[i], "-") == 0){
                int *counted;
                counted = counter(stdin);
                for(k = 0; k < 3; k++){
                    total[k] += counted[k];
                }
                if(l == 'l')
                    printf("\t%d", counted[0]);
                if(w == 'w')
                    printf("\t%d", counted[1]);
                if(c == 'c') {
                    printf("\t%d", counted[2]);
                }
                printf(" %s\n", "-");
            }
            else{
                if(strchr(argv[i], '-'))
                    continue;
                FILE* file = fopen(argv[i], "r");
                if(file == NULL) {
                    fprintf(stderr, "%s : %s\n", argv[i], strerror(errno));
                    return 1;
                }
                struct stat checker;

                if(stat(argv[i], &checker) < 0 ) {
                    return 2;
                }

                if(checker.st_mode & S_IRUSR) {
                }
                else{
                    fprintf(stderr, "%s : %s\n", argv[i],strerror(EACCES));
                    return 3;
                }

                int *counted;
                counted = counter(file);

                for(k = 0; k < 3; k++){
                    total[k] += counted[k];
                }

                if(l == 'l')
                    printf("\t%d", counted[0]);
                if(w == 'w')
                    printf("\t%d", counted[1]);
                if(c == 'c') {
                    printf("\t%d", counted[2]);
                }
                printf(" %s\n", argv[i]);
            }
        }
        if(l == 'l')
            printf("\t%d", total[0]);
        if(w == 'w')
            printf("\t%d", total[1]);
        if(c == 'c') {
            printf("\t%d", total[2]);
        }
        printf(" total\n");
    }
    return 0;
}

4 个答案:

答案 0 :(得分:1)

它适用printf的原因是巧合:你的程序有未定义的行为,因为你正在返回一个指向本地的指针。分配给局部变量的内存在从函数返回时被重用,因此在函数外部引用该内存会导致访问垃圾值。但是,有时候,由于调用者使用堆栈的方式,特定地点不会被重用。在这种情况下,该程序给出了工作的外观,但即使对代码进行少量更改也可能导致产生错误的结果甚至崩溃。

按如下方式更改您的代码:

void counter(FILE* file, int count[]) {
    int nsymbols = 0, nwords = 0, nlines = 0;
    char c;
    while (EOF != (c = fgetc(file))) {
         nsymbols++;
         if(c == ' '){
            nwords++;
         }
         if(c == '\n'){
            nlines++;
            nwords++;
         }
      }
      count[0] = nlines;
      count[1] = nwords;
      count[2] = nsymbols;
}

并像这样调用此函数:

int counted[3];
counter(file, counted);

这将解决问题。

答案 1 :(得分:0)

此处count数组

  int count[] = {nlines, nwords, nsymbols};
  return count;

超出范围,其生命周期在函数返回后结束。您可以将其设置为静态或malloc内存,这两者都将延长指向超出函数执行时间的内存的生命周期。

答案 2 :(得分:0)

问题是你要返回一个本地数组的数组count。你不能这样做。我将是未定义的行为。一切都会发生!

似乎数组的大小是固定的,所以你应该声明它并将它作为参数传递给函数counter()。

答案 3 :(得分:0)

使数组静态或通过malloc或calloc使用动态内存分配创建它...因为在堆中创建的数组一直保持到程序结束