如何使用fopen

时间:2019-05-20 18:11:09

标签: c fopen fgets fgetc feof

使用fgets时,我想忽略/跳过文本文件中的注释。

问题是,如果行首的第一个字符是#,我只能跳过注释。注释以文本文件中的#开头。但是我的#中有一些file.txt不是一行的第一个字符,就像这样;

#Paths
A B #Path between A and B.
D C #Path between C and D.

A是我的第一个节点,B是我的第二个节点,当#出现时,我想忽略其余文本,直到下一行。我的新节点应该是D和C等。我只能在fopen函数中使用“ r”。 我已经尝试过fgets,但它逐行读取,fgetc也无济于事。

    bool ignore_comments(const char *s)
    {
        int i = 0;
        while (s[i] && isspace(s[i])) i++;
        return (i >= 0 && s[i] == '#');
    }
    FILE *file;
    char ch[BUFSIZE];
    file = fopen("e.txt", "r");
    if (file == NULL) {
        printf("Error\n");
        fprintf(stderr, "ERROR: No file input\n");
        exit(EXIT_FAILURE);
    }
    while(fgets(ch, BUFSIZE, file) != NULL)
    {
              if (line_is_comment(ch)) {
                        // Ignore comment lines.
                        continue;
                printf("%c",*ch);
                }
     fscanf(file, "%40[0-9a-zA-Z]s", ch);
....
}

3 个答案:

答案 0 :(得分:0)

方法名称也不同,但是我对这个版本正确吗? 忽略我的脏方法line_is_comment-从第一个版本开始,除非您想使用;-)

扩展的测试输入:

#Paths
A B #Path between A and B.
D C #Path between C and D.
E F
G H

输出:

 rest of line read
AB rest of line read
DC rest of line read
EF rest of line read
GH rest of line read
#include <stdio.h>

bool line_is_comment(const char *s)
{
    char *commentPos = const_cast<char*>(strchr(s, '#'));
    if(commentPos != NULL) {
        *commentPos = 0; // cut-off chars after comment
        //return true; // or false then to accept the line
        return commentPos == s;
    }
    return false;
}

#define BUFSIZE 50

int main()
{
    FILE *file;
    char ch[BUFSIZE];
    file = fopen("e.txt", "r");
    if (file == NULL) {
        printf("Error\n");
        fprintf(stderr, "ERROR: No file input\n");
        exit(EXIT_FAILURE);
    }
    int x;
    while(!feof(file)) {
        x = fscanf(file, "%40[0-9a-zA-Z]s", ch);
        if(x == 0) {
            ch[0] = fgetc(file);
            if(ch[0] == '#' || ch[0] == '\n') {
                if(ch[0] != '\n') fgets(ch, BUFSIZE, file);
                printf(" rest of line read\n");
            }
        } else if(x<0) break;
        else {
                 printf("%c",*ch); // continue with ... undisclosed part here
            }
    }

    return 0;
}

答案 1 :(得分:0)

以下建议的代码:

  1. 执行所需的功能
  2. 干净地编译
  3. 正确检查错误
  4. 此答案使用状态机,基于:'InComment'

现在,建议的代码:

#include <stdio.h>
#include <stdlib.h>

int main( void )
{
    int InComment = 0;

    FILE *fp = fopen( "file.txt", "r" );
    if( !fp )
    {
        perror( "fopen to read -file.txt- failed" );
        exit( EXIT_FAILURE );
    }

    int ch;

    while( (ch = fgetc(fp)) != EOF )
    {
        if( ch == '#' )
        {
            InComment = 1;
        }

        else if( ch == '\n' )
        {
            InComment = 0;
            fputc( ch, stdout );
        }

        else if( !InComment )
        {
            fputc( ch, stdout );
        }
    }
    fclose( fp );
}

答案 2 :(得分:0)

您还可以在单​​个简单调用中使用strcspn修剪所有注释(如果不存在,则修剪缓冲区中的行尾)。通常在fgets()读取的缓冲区中修剪行尾的位置:

        ch[strcspn (ch, "\r\n")] = 0;  /* trim line-ending */

您可以简单地将"#"字符添加到您的拒绝列表中,如果存在评论,则可以在其中 nul-terminate 。这样可以减少删除以'#'开头的注释并将新格式化的行输出到以下位置的完整任务:

    while (fgets (ch, BUFSIZE, fp)) {   /* read every line */
        ch[strcspn (ch, "#\r\n")] = 0;  /* trim comment or line-ending */
        puts (ch);                      /* output line w/o comment */
    }

一个简短的示例,将文件读取为程序的第一个参数(或者,如果没有给出参数,则默认从stdin读取),您可以这样做:

#include <stdio.h>
#include <string.h>

#define BUFSIZE 1024    /* if you need a constant, #define one (or more) */

int main (int argc, char **argv) {

    char ch[BUFSIZE];
    /* use filename provided as 1st argument (stdin by default) */
    FILE *fp = argc > 1 ? fopen (argv[1], "r") : stdin;

    if (!fp) {  /* validate file open for reading */
        perror ("file open failed");
        return 1;
    }

    while (fgets (ch, BUFSIZE, fp)) {   /* read every line */
        ch[strcspn (ch, "#\r\n")] = 0;  /* trim comment or line-ending */
        puts (ch);                      /* output line w/o comment */
    }

    if (fp != stdin) fclose (fp);       /* close file if not stdin */

    return 0;
}

示例输入文件

借用汤姆的示例文件:)

$ cat dat/comments_file.txt
#Paths
A B #Path between A and B.
D C #Path between C and D.
E F
G H

使用/输出示例

$ ./bin/comments_remove <dat/comments_file.txt

A B
D C
E F
G H

仔细检查一下,如果还有其他问题,请告诉我。