繁体   English   中英

Fscanf读取c语言中float和character的混合

[英]Fscanf to read a mixture of float and characters in c

所以我有一个.txt文件,如下所示:

** Paris ** Flight,5 days,visiting various monuments. | 2999.99 |
** Amsterdam ** By bus,7 days, local art gallery. | 999.99 |
** London ** Flight,3 days,lots of free time. | 1499.99 |

我想将信息存储到3个变量中,即城市,描述和价格,但根本无法保存行。

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

int main()
{
    FILE *fp;
    char city[256],desciption[256];
    float temp;
    if(fp=fopen("Ponuda.txt","rt")==NULL){
        printf("ERROR\n");
        exit(1);
    }

    while(fscanf(fp,"** %s ** %s | %f |",city,description,&temp)==3){

        printf("** %s ** %s |%f|\n",city,description,temp);
    }
    return 0;

在IMO中,使用fgets读取每个文件行要容易得多,然后使用strtok用分隔符字符串"*|"分隔每一行。

然后,我使用strdup将文本字符串复制到struct中,并使用sscanf从第三个令牌中提取票价。 我应该已经测试了strdup的返回值,因为它在内部调用了malloc

我还使用double而不是float (仅在有约束的情况下使用)。

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

#define MAX  10

typedef struct {
    char *city;
    char *descrip;
    double fare;
} flight_t;

int main()
{
    FILE *fp;
    flight_t visit[MAX] = {0};
    char buffer [1024];
    char *tok;
    int records = 0, i;
    if((fp = fopen("Ponuda.txt", "rt")) == NULL) {
        printf("Error opening file\n");
        exit(1);
    }

    while(fgets(buffer, sizeof buffer, fp) != NULL) {
        if((tok = strtok(buffer, "|*")) == NULL) {
            break;
        }
        if(records >= MAX) {
            printf("Too many records\n");
            exit(1);
        }
        visit[records].city = strdup(tok);          // add NULL error checking

        if((tok = strtok(NULL, "|*")) == NULL) {    // pass NULL this time
            break;
        }
        visit[records].descrip = strdup(tok);       // add NULL error checking

        if((tok = strtok(NULL, "|*")) == NULL) {    // pass NULL this time
            break;
        }
        if(sscanf(tok, "%lf", &visit[records].fare) != 1) {     // read a double
            break;
        }
        records++;
    }

    fclose(fp);

    // print the records
    for(i = 0; i < records; i++) {
        printf("** %s ** %s |%.2f|\n", visit[i].city, visit[i].descrip, visit[i].fare);
    }

    // free the memory given by strdup
    for(i = 0; i < records; i++) {
        free(visit[i].city);
        free(visit[i].descrip);
    }
    return 0;
}

程序输出:

**  Paris  **  Flight,5 days,visiting various monuments.  |2999.990000|
**  Amsterdam  **  By bus,7 days, local art gallery.  |999.990000|
**  London  **  Flight,3 days,lots of free time.  |1499.990000|

用单个fscanf语句很难做到这一点,因为正如@Barmar指出的那样,城市名称可能超过一个单词。 阅读整行(使用fgets)然后自己解析该行要容易得多。 要解析它:

  1. 在“ **”之后找到第一个非空白字符
  2. 找到此后的最后一个非空白字符,该字符之前应为“ **”
  3. 这些索引之间的文本将是城市名称(可能包含空格)
  4. 描述是在步骤2中找到的“ **”和下一个“ |”的索引之间的文本。 之后。
  5. 成本是“ |”之后的字符串

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM