逐字阅读文件并输出白色空格

时间:2015-11-03 02:07:58

标签: c file io

输入文本文件如下:

Hello my
name is 
mark.
and
im 
going
to
love
c!

代码:

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

 int main(int argc, char *argv[]){
FILE *pFile;
char x[60];

pFile = fopen("test0.txt","r");

if(pFile != NULL){
    while(fscanf(pFile, " %60s", x) == 1){
        printf("%s",x);
 }

}
}

输出文字文件为:

Hellomynameismark.andimgoingtolovec!

我希望输出如下:

Hello my name is mark. and im going to love c!

非常新的C程序员所以只知道基础知识。

编辑----

   int main(int argc, char *argv[]){
FILE *pFile;
char x[60],line[60];

pFile = fopen("test0.txt","r");

while(!feof(pFile)){

        fgets(line, 60, pFile);
        strtok(line, "\r\n");
        printf("%s", line );
    }
    fclose(pFile);

输出:

 Hello myname is mark.andim goingtolovec!

这不会在新行之间留下空格。但是,如果我取出strtok行,输出将是这样的:

Hello my
name is 
mark.
and
im 
going
to
love
c!

- 编辑

 .sp 2
 .ce
 This is an example file for formatting.
 .sp 1
 The above line
 was formatted using a .ce 1 command, which means 'centre
 the following line',
 so it should appear in the 
 centre of the page.
 The paragraph was separated from the heading using
 a .sp 1 command to create a single blank line.
 There should also be two blank lines above the centred heading to make            reading it slightly easier.

3 个答案:

答案 0 :(得分:4)

简单的答案是:

while(fscanf(pFile, " %59[^\n]%*c", x) == 1)

此处%[^\n]使用字符类[stuff]来读取换行符之前的所有内容。 %*c只需读取并丢弃换行符,而不将其添加到fscanf的匹配计数中。

但是对于面向行的输入,您应该使用标准库提供的面向行的函数之一(例如fgets或POSIX { {1}})。

使用fgets&amp; strtok的

正如你从评论中得到的那样,使用getline会让你感到悲伤。您只需使用feof的返回来确定文件结尾。这是一个将拼图的所有部分放在一起的例子:

fgets

<强>编译

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

#define MAXWDS  20
#define MAXCHR  60

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

    char line[MAXCHR] = {0};
    char *words[MAXWDS] = {NULL};
    FILE *pFile = NULL;
    size_t i, index = 0;

    /* open file for reading (if provided), or read from stdin */    
    if (!(pFile = argc > 1 ? fopen (argv[1], "r") : stdin)) {
        fprintf (stderr, "error: file open failed '%s'.\n", argv[1]);
        return 1;
    }

    while (fgets (line, 60, pFile)) 
    {
        char *p = line;

        /* split line into tokens, stored in words[] */
        for (p = strtok (p, " \r\n"); p; p = strtok (NULL, " \r\n")) {
            words[index++] = strdup (p); /* allocate & copy */

            if (index == MAXWDS)    /* check pointer limit */
                break;
        }

    }
    if (pFile != stdin) fclose (pFile);

    /* output in a single line */
    for (i = 0; i < index; i++) {
        printf (" %s", words[i]);
        free (words[i]);            /* free allocated memory */
    }

    putchar ('\n');

    return 0;
}

<强>输出

gcc -Wall -Wextra -o bin/fgets_strtok fgets_strtok.c 

注意:只需在单词之间打印出空格,只要每行中每个单词之间已经有空格,就没有理由去麻烦要将每一行分成单独的单词,您可以简单地以空格分开的方式打印每行的内容。您使用$ ./bin/fgets_strtok dat/hellomark.txt Hello my name is mark. and im going to love c! 遇到的唯一问题是,它还会将fgets(或newline)作为字符串的一部分进行读取。这很容易删除。您可以用以下内容替换整个读取循环:

carriage return, newline

<强>输出

while (fgets (line, 60, pFile)) 
{
    size_t len = strlen (line);

    /* strip trailing newline (or carriage return newline ) */
    while (len && (line[len-1] == '\n' || line[len-1] == '\r'))
        line[--len] = 0;  /* overwrite with null-terminating char */

    words[index++] = strdup (line); /* allocate & copy */

    if (index == MAXWDS)    /* check pointer limit */
        break;
}

从文件中读取的标准方式(不是文件或标准输入)

通过包含打开文件(如果在命令行中提供)或从$ ./bin/fgets_mark <dat/hellomark.txt Hello my name is mark. and im going to love c! 读取(如果没有提供文件名)的方法,我为您提前了解道歉。标准方法是首先检查命令行上是否提供了正确数量的参数,然后打开提供的文件名,验证它是否已打开,然后处理输入。我所做的是在stdin命令中抛出一个ternary operator

fopen

pFile = argc > 1 ? fopen (argv[1], "r") : stdin 符号的右侧是三元运算符,它只是'='的简写。它的作用是问if -> then -> else?如果测试argc > 1,则true。如果pFile = fopen (argv[1], "r");测试argc > 1,则false

看看标准方式是否更有意义:

pFile = stdin;

答案 1 :(得分:1)

一个简单的状态机可以解决问题 - 没有行长度限制。

#include <stdio.h>

 int main(void) {
   FILE *pFile = fopen("test0.txt","r");
   if(pFile != NULL) {
     int previous_isspace = 0;
     int ch;
     for (;;) {
       ch = fgetc(pFile);
       if (ch == EOF) break;
       if (isspace(ch)) {
         previous_isspace = 1;
       } else {
         if (previous_isspace == 1) {
           fputc(' ', stdout);
         }
         previous_isspace = 0;
         fputc(ch, stdout);
       }  
     }
     fclose(pFile);
     fputc('\n', stdout);   // If code should have a \n at the end
   }
 }

答案 2 :(得分:0)

我认为,只要看看我就错过任何东西就足够了。

if(pFile != NULL){
      //  while(fscanf(pFile, " %60s", x) == 1){
            while (fgets(x, sizeof(x), pFile) != NULL) {
             token = strtok(x,"\r\n");
             if(token != NULL)
                    printf("%s ",x);
             else
                    printf("%s",x);
            }
            fclose(pFile);
    }