gpt4 book ai didi

c - 跳过特定行并将变量特定变量存储在C中

转载 作者:太空宇宙 更新时间:2023-11-04 03:22:48 25 4
gpt4 key购买 nike

这是一个简单的问题,但我似乎想不通。我需要一点帮助。我有一个名为programFile的文件,如下所示:

start mul val1
<--tab-->ldb #2
<--tab-->addr A B
<--tab-->float
loop lda #1
<--tab-->sta val2
<--tab-->j loop
val1 word 12
val2 word 0

我只需要不以制表符开头的行,然后对它们执行一些操作(目前我只想打印该行中的第一个单词),因此只有行 start mul val1loop lda #1val1 word 12val2 word 0。输出应为:
start
loop
val1
val2

最后,我还想从标签开始,做一些不同的事情,但这是我解决问题的拙劣尝试:
while(ch = fgetc(programFile) != EOF){
if(ch == '\t'){
while(ch != '\n'){
ch = fgetc(programFile);
}
}else{
fscanf(programFile, "%s", symbol);
printf("%s\n", symbol);
}
}

这是我的输出:
tart
ldb
addr
float
loop
sta
j
val1
val2

最佳答案

从注释继续,虽然使用面向字符的输入函数(例如fgetcgetc)没有错,但是当需要处理数据的“行”时,最好使用C-library(fgets)或POSIX(getline)提供的面向行的输入函数,然后从每一行解析所需的信息。
为什么?基本上是方便和高效。面向行的输入函数提供一个缓冲读取(每次读取多个字符),对于大型输入文件,它确实有助于文件I/O,您将读取整行(如果使用fgets时有足够的行存储空间,则读取多个块,直到读取整行;getline将自动分配(和重新分配)足够的存储空间来容纳每行)。
然后,您将拥有sscanfstrtokstrsepstrstrstrchrline[0]等工具。。您可以从存储行中解析所需的任何内容。(您也可以始终使用简单的指针算法解析带有指针或指针对的任何行,“遍历字符串”并在运行时比较每个字符)在内存中对存储字符串中的每个字符执行的操作比在同时对每个字符执行文件I/O时执行同一操作快几个数量级。
当您关心每一行的起始字符时,只需要将*line(或简单地stdin)与您要查找的任何字符进行比较。
下面是一个简单的示例,它从作为第一个参数给定的输入文件名(如果没有给定文件名,则默认从tab)读取,然后测试每行的开头字符。如果行以tab开头,它只输出前面有tab的行(在通过输出line + 1跳过文件中的-- begins with tab之后),后面跟着-- no tab(您可以随意处理这些行,或者完全跳过它们),否则它将输出行本身,后面跟着'\n'。不同前缀行的处理完全取决于您。如果需要,可以构建一个指针数组,其中包含每种不同类型的行,或者使用包含指针数组中的命令和选项卡内容的结构来保留行关联(哪些命令与哪些选项卡行一起使用)。
在面向行的输入函数中,唯一需要注意的是,它们可以读取并包含尾随的newlines。您通常不希望存储结尾处悬挂着newlines的字符串,因此您将希望用nul终止字符覆盖尾随的'\n'来修剪strlen。此示例通过使用newline获取每行的长度,然后使用0覆盖'\0'(相当于字符tab)。我不喜欢打字。。。

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

#define MAX 64

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

char line[MAX] = "";
FILE *fp = argc > 1 ? fopen (argv[1], "r") : stdin;

if (!fp) { /* validate file open for reading */
fprintf (stderr, "error: file open failed '%s'.\n", argv[1]);
return 1;
}

while (fgets (line, MAX, fp)) /* read each line in file */
{
size_t len = strlen (line); /* get the length */

if (line[len - 1] == '\n') /* check for trailing '\n' */
line[--len] = 0; /* overwrite with nul-byte */

if (*line == '\t') { /* if first char is '\t' */
printf ("\t%s - begins with tab\n", line + 1);
continue;
}

printf ("%s - no tab\n", line); /* line has no tab */
}

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

return 0;
}

输入文件
$ cat dat/tabfile.txt
start mul val1
ldb #2
addr A B
float
loop lda #1
sta val2
j loop
val1 word 12
val2 word 0

示例使用/输出
$ ./bin/filehandletab <dat/tabfile.txt
start mul val1 - no tab
ldb #2 - begins with tab
addr A B - begins with tab
float - begins with tab
loop lda #1 - no tab
sta val2 - begins with tab
j loop - begins with tab
val1 word 12 - no tab
val2 word 0 - no tab

如注释中所指出的,如果您的意图是从不是以a strchr开头的行中解析第一个单词,那么您可以简单地使用 space来定位第一个 space,临时在 space处终止行,使用该命令,然后还原 if...else...,以便在需要时可以进一步解析字符串,例如。
while (fgets (line, MAX, fp)) 
{
char *p = NULL;
size_t len = strlen (line);
...
if (*line == '\t') { /* if first char is '\t' */
printf ("\t%s - begins with tab\n", line + 1);
continue;
}

if ((p = strchr (line, ' '))) { /* find first ' ' */
*p = 0; /* terminate at p */
printf ("%s - no tab\n", line); /* output line */
*p = ' '; /* restore ' ' */
}
else
printf ("%s - no tab\n", line); /* s has no tab */
}

或者,编写相同的行终止符,删除 printf和重复的 ,您可以在更紧凑但不太可读的代码中执行以下操作(完全由您决定):
    if ((p = strchr (line, ' ')))        /* find first ' ' */
*p = 0; /* terminate at p */

printf ("%s - no tab\n", line); /* s has no tab */

if (p) /* if terminated */
*p = ' '; /* restore ' ' */

示例使用/输出
$ ./bin/filehandletab <dat/tabfile.txt
start - no tab
ldb #2 - begins with tab
addr A B - begins with tab
float - begins with tab
loop - no tab
sta val2 - begins with tab
j loop - begins with tab
val1 - no tab
val2 - no tab

再看一遍,如果你还有什么问题,请告诉我。

关于c - 跳过特定行并将变量特定变量存储在C中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43903249/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com