使用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);
....
}
答案 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)
以下建议的代码:
现在,建议的代码:
#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
仔细检查一下,如果还有其他问题,请告诉我。