c - 无法弄清楚如何隔离和替换数字

标签 c scanf readfile data-manipulation fgetc

我所做的是使用第一组代码提取了数据并将其放入新文件中,因此它只是值

这是我输入的第一组代码:

original input

这是第二组代码的输入文件,这是我的主要问题:

second set of input code

然后在第二组代码中读取新文件的位置,以便使用fgetc输出数字,我现在如何使用此格式应用简单的公式?
具体来说,公式是将任何10的值变成0的值
我尝试过,并且由于fgetc是unsigned char,所以我尝试在if语句中使用10的二进制数

到目前为止,这是我的代码:

#include<stdio.h> 

int main() 
{ 
    FILE* ptr = fopen("Data.txt","r"); 
    if (ptr==NULL) 
    { 
        printf("no such file."); 
        return 0; 
    } 

    FILE*fp = fopen("/data flow/NEWdata.txt", "w+");

        int x;
  int count=0;
    char buf[100];
    fscanf(ptr,"%*s %*s %*s %*s %*s %*s %*s %s ",buf); //skip first line and stuff before first value(column names)
    while (fscanf(ptr,"%*s %*s %*s %*s %s  ",buf)==1) 
    {

        fprintf(fp, "%s\n",buf);
    }
    fclose(fp); 
    return 0; 
} 


第二套代码

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

int main () {
   FILE *fp;
   char str[100];

   /* opening file for reading */
   fp = fopen("NEWdata.txt" , "r");
   if(fp == NULL) {
      perror("Error opening file");
      return(-1);
   }
   int i;
   while( fgets (str, 100, fp)!=NULL ) {
      /* writing content to stdout */
      sscanf (str,%u,i);
    printf(%u,i);

   }
   fclose(fp);

   return(0);
}


我的命令行
enter image description here

错误我正在下面
enter image description here

 if (measure = 10)
            fprintf (ofp, "%f\n", measure - 10);
        else
            fprintf (ofp, "%f\n", measure);


#include <stdio.h>

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

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

    char buf[MAXC];     /* buffer to hold each line read from file */
    size_t n = 0;       /* line counter */
   FILE *fp, *ofp;     /* file pointer, output file pointer */

    if (argc < 3 ) {    /* validate 2 arguments given for in/out filenames */
        fprintf (stderr, "error: insufficient input,\n"
                         "usage: %s filename outfilename\n", argv[0]);
        return 1;
    }

    /* open file/validate file open for reading */
    if ((fp = fopen ("Data.txt", "r")) == NULL) {
        perror ("fopen-argv[1]");
        return 1;
    }

    if ((ofp = fopen ("NEWdata.txt", "w")) == NULL) {
        perror ("fopen-argv[2]");
        return 1;
    }

    if (!fgets (buf, MAXC, fp)) {   /* read/discard 1st line */
        fputs ("error: EOF on read of first line.\n", stderr);
        return 1;
    }

    while (fgets (buf, MAXC, fp)) { /* read all remaining lines */
        char date[MAXF], time[MAXF], prod[MAXF], unit[MAXF];
        double measure;

        if (sscanf (buf, "%s %s %s %lf %s", /* validate all 5 fields */
                    date, time, prod, &measure, unit) != 5) {
            fprintf (stderr, "error: invalid format, line: %zu\n", n + 1);
            continue;
        }

        /* do whatever you need with the separated values.
         * if measure is 10.xx, then make measure 0.xx
         */
        if (measure = 10)
            fprintf (ofp, "%f\n", measure - 10);
        else
            fprintf (ofp, "%f\n", measure);

        n++;    /* increment line counter */
    }
}

最佳答案

如果我理解正确,并且您想从Data.txt读取值,并且如果Measurement的值是10,则要将值设置为0。如我的评论所述,您的Measurement值是一个浮点数(下面使用double)。由于不能完全以浮点格式表示所有数字,因此浮点数的比较本质上是不精确的。 (可以精确表示10.0,但要注意限制)。请参见Is floating point math broken?Why Are Floating Point Numbers Inaccurate?

在到达那里之前,请避免对文件名进行硬编码或在代码中使用幻数。 main()带有允许您将信息传递给代码的参数。您可以简单地将文件名传递为程序的第一个参数,也可以将文件名作为输入。例如

#include <stdio.h>

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

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

    char buf[MAXC];     /* buffer to hold each line read from file */
    size_t n = 0;       /* line counter */
    FILE *fp;           /* file pointer */

    if (argc < 2 ) {    /* validate 1 argument given for filename */
        fprintf (stderr, "error: insufficient input,\n"
                         "usage: %s filename\n", argv[0]);
        return 1;
    }

    /* open file/validate file open for reading */
    if ((fp = fopen (argv[1], "r")) == NULL) {
        perror ("fopen-argv[1]");
        return 1;
    }


使用fgets()和缓冲区(例如,MAXC的字符数组,最大字符),您可以读取和丢弃第一行。 (*不要忘记验证读取的内容):

    if (!fgets (buf, MAXC, fp)) {   /* read/discard 1st line */
        fputs ("error: EOF on read of first line.\n", stderr);
        return 1;
    }


现在只需循环将每个带有fgets()的行读入同一缓冲区中,然后将该缓冲区传递给sscanf()即可从该行中解析所需的值,例如

    while (fgets (buf, MAXC, fp)) { /* read all remaining lines */
        char date[MAXF], time[MAXF], prod[MAXF], unit[MAXF];
        double measure;

        if (sscanf (buf, "%s %s %s %lf %s", /* validate all 5 fields */
                    date, time, prod, &measure, unit) != 5) {
            fprintf (stderr, "error: invalid format, line: %zu\n", n + 1);
            continue;
        }


(请注意:如果该行的格式不正确,则您的读取不会像使用fscanf()那样失败。这里,通过使用fgets()进行读取,您一次可以消耗一行,而不管格式如何,然后使用< cc>来解析该行中的值。如果该行格式与您的format-string不匹配,则只需丢弃该行并读取下一行。)

现在,您可以使用分隔为sscanf()的值执行任何操作。在这里,我们将简单地比较date, time, prod, measure & unit以查看它是否等于或大于等于measure(不包括在内),并从使10.0变为11.0的值中减去10.0。您可以根据需要简单地输出0.xxx

        /* do whatever you need with the separated values.
         * if measure is 10.xx, then make measure 0.xx
         */
        if (10.0 <= measure && measure < 11.0)
            printf ("measurement[%2zu]: %.3f   (was %.3f)\n",
                    n, measure - 10., measure);
        else
            printf ("measurement[%2zu]: %.3f\n", n, measure);

        n++;    /* increment line counter */
    }
}


这就是您想要的一个简单示例。 (1)用10.xxx读取和(2)用0.0解析的基本原理可以涵盖各种输入情况。您将反复使用它。这样做而不是直接使用fgets()的原因是为了防止出现单行格式错误,从而导致匹配失败,在这种情况下,从输入流中提取字符会停止使输入流中有问题的字符保持未读状态,而只是等待在您下一次尝试输入时再次咬您。

完整的示例是:

#include <stdio.h>

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

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

    char buf[MAXC];     /* buffer to hold each line read from file */
    size_t n = 0;       /* line counter */
    FILE *fp;           /* file pointer */

    if (argc < 2 ) {    /* validate 1 argument given for filename */
        fprintf (stderr, "error: insufficient input,\n"
                         "usage: %s filename\n", argv[0]);
        return 1;
    }

    /* open file/validate file open for reading */
    if ((fp = fopen (argv[1], "r")) == NULL) {
        perror ("fopen-argv[1]");
        return 1;
    }

    if (!fgets (buf, MAXC, fp)) {   /* read/discard 1st line */
        fputs ("error: EOF on read of first line.\n", stderr);
        return 1;
    }

    while (fgets (buf, MAXC, fp)) { /* read all remaining lines */
        char date[MAXF], time[MAXF], prod[MAXF], unit[MAXF];
        double measure;

        if (sscanf (buf, "%s %s %s %lf %s", /* validate all 5 fields */
                    date, time, prod, &measure, unit) != 5) {
            fprintf (stderr, "error: invalid format, line: %zu\n", n + 1);
            continue;
        }

        /* do whatever you need with the separated values.
         * if measure is 10.xx, then make measure 0.xx
         */
        if (10.0 <= measure && measure < 11.0)
            printf ("measurement[%2zu]: %.3f   (was %.3f)\n",
                    n, measure - 10., measure);
        else
            printf ("measurement[%2zu]: %.3f\n", n, measure);

        n++;    /* increment line counter */
    }
}


输入文件示例

$ cat dat/data_timestamp.txt
TimeDateStamp Product Measurement Unit
2019-11-09 16:54    FS2012  0.344   SLPM
2019-11-09 16:54    FS2012  0.344   SLPM
2019-11-09 16:54    FS2012  0.344   SLPM
2019-11-09 16:54    FS2012  0.344   SLPM
2019-11-09 16:54    FS2012  0.136   SLPM
2019-11-09 16:54    FS2012  0.136   SLPM
2019-11-09 16:54    FS2012  0.136   SLPM
2019-11-09 16:54    FS2012  0.136   SLPM
2019-11-09 16:54    FS2012  0.047   SLPM
2019-11-09 16:54    FS2012  0.047   SLPM
2019-11-09 16:54    FS2012  0.047   SLPM
2019-11-09 16:54    FS2012  0.047   SLPM
2019-11-09 16:54    FS2012  1.991   SLPM
2019-11-09 16:54    FS2012  1.991   SLPM
2019-11-09 16:54    FS2012  1.991   SLPM
2019-11-09 16:54    FS2012  10.0    SLPM
2019-11-09 16:54    FS2012  10.661  SLPM
2019-11-09 16:54    FS2012  10.991  SLPM
2019-11-09 16:54    FS2012  11.0    SLPM


使用/输出示例

下面,我显示sscanf()减少到fscanf()的位置:

$ ./bin/read_data_timestamp dat/data_timestamp.txt
measurement[ 0]: 0.344
measurement[ 1]: 0.344
measurement[ 2]: 0.344
measurement[ 3]: 0.344
measurement[ 4]: 0.136
measurement[ 5]: 0.136
measurement[ 6]: 0.136
measurement[ 7]: 0.136
measurement[ 8]: 0.047
measurement[ 9]: 0.047
measurement[10]: 0.047
measurement[11]: 0.047
measurement[12]: 1.991
measurement[13]: 1.991
measurement[14]: 1.991
measurement[15]: 0.000   (was 10.000)
measurement[16]: 0.661   (was 10.661)
measurement[17]: 0.991   (was 10.991)
measurement[18]: 11.000


查看情况,让我知道这是否接近您的需求。我可能仍会误解您想从10转换为0所要实现的目标,但是如果我误解了并且很乐意提供进一步的帮助,请在下面留言。

写入输出文件作为程序的第二个论点

要将调整后的10值单独写入所选的输出文件,您只需在命令行中输入文件名后提供要使用的输出文件名,然后使用0打开文件进行输出(第二个Measurement的命令行参数)更改很小。只需添加另一个argv[2]指针main()(用于输出文件指针),然后打开该文件以进行写入:

    FILE *fp, *ofp;     /* file pointer, output file pointer */

    if (argc < 3 ) {    /* validate 2 arguments given for in/out filenames */
        fprintf (stderr, "error: insufficient input,\n"
                         "usage: %s filename outfilename\n", argv[0]);
        return 1;
    }

    /* open file/validate file open for reading */
    if ((fp = fopen (argv[1], "r")) == NULL) {
        perror ("fopen-argv[1]");
        return 1;
    }

    if ((ofp = fopen (argv[2], "w")) == NULL) {
        perror ("fopen-argv[2]");
        return 1;
    }


然后只需使用FILE*输出调整后的测量值,例如

        if (10.0 <= measure && measure < 11.0)
            fprintf (ofp, "%.3f\n", measure - 10.);
        else
            fprintf (ofp, "%.3f\n", measure);


(程序的其余部分完全相同)

#include <stdio.h>

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

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

    char buf[MAXC];     /* buffer to hold each line read from file */
    size_t n = 0;       /* line counter */
    FILE *fp, *ofp;     /* file pointer, output file pointer */

    if (argc < 3 ) {    /* validate 2 arguments given for in/out filenames */
        fprintf (stderr, "error: insufficient input,\n"
                         "usage: %s filename outfilename\n", argv[0]);
        return 1;
    }

    /* open file/validate file open for reading */
    if ((fp = fopen (argv[1], "r")) == NULL) {
        perror ("fopen-argv[1]");
        return 1;
    }

    if ((ofp = fopen (argv[2], "w")) == NULL) {
        perror ("fopen-argv[2]");
        return 1;
    }

    if (!fgets (buf, MAXC, fp)) {   /* read/discard 1st line */
        fputs ("error: EOF on read of first line.\n", stderr);
        return 1;
    }

    while (fgets (buf, MAXC, fp)) { /* read all remaining lines */
        char date[MAXF], time[MAXF], prod[MAXF], unit[MAXF];
        double measure;

        if (sscanf (buf, "%s %s %s %lf %s", /* validate all 5 fields */
                    date, time, prod, &measure, unit) != 5) {
            fprintf (stderr, "error: invalid format, line: %zu\n", n + 1);
            continue;
        }

        /* do whatever you need with the separated values.
         * if measure is 10.xx, then make measure 0.xx
         */
        if (10.0 <= measure && measure < 11.0)
            fprintf (ofp, "%.3f\n", measure - 10.);
        else
            fprintf (ofp, "%.3f\n", measure);

        n++;    /* increment line counter */
    }
}


然后如果我使用两个参数(例如输入和输出文件名)调用程序,如下所示:

$ ./bin/read_data_timestamp2 dat/data_timestamp.txt dat/measure_val.txt


我将从我的输入文件ofp中读取并将结果写入我的输出文件fprintf中(您可以将任意文件名用作输出文件,但您的输入将是dat/data_timestamp.txt)。

结果输出文件

$ cat dat/measure_val.txt
0.344
0.344
0.344
0.344
0.136
0.136
0.136
0.136
0.047
0.047
0.047
0.047
1.991
1.991
1.991
0.000
0.661
0.991
11.000


让我知道是否可以解决。

关于c - 无法弄清楚如何隔离和替换数字,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59095995/

相关文章:

C 编程程序在未询问时期望输入

c - scanf 成一个字符串数组

javascript - 使用 php 下载 .js 文件

php - 试图生成一个 php 下载链接。文件名不会设置

winapi - ReadFile 在从子进程结束后读取 stdout 时不返回

ios - 未缓存模块时,iOS 框架中的 C 静态库“无法构建 Objective-C 模块”

从加载的共享对象库中调用主程序函数

mysql - 将 Double 类型的数据直接从 C 发送到 MySQL

c - 将 int 存储到 char 数组并从中检索

c - 无效的转换标识符 '.'