- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个我正在尝试编写的 C 程序,它打算反转文件的行。我在 C 方面仍然很无能(尽管我来自 Java 背景),所以我很可能会在指针等方面犯错误,但我已经尝试每次都查阅手册。这有点像是一项任务。
程序的要点是反转文件的内容,最多 MAX_LINES 行,每行不超过 MAX_CHARS。我想尝试的方法如下:使用 fgets 从文件中读取 80 个字符或直到 EOL,存储该字符串,然后重复该过程直到达到 EOF 或 MAX_LINES,使用旁边的计数器。之后我只是将相同的字符串放在不同的数组中,从 second_array[counter] 到 0。但是,我在实际将字符串放入第一个数组时遇到了问题。这是我到目前为止所拥有的:
1 #include <stdio.h>
2 #include <string.h>
3
4 #define MAX_LINES 100
5 #define MAX_CHAR 80
6
7 int main(int argc, char *argv[])
8 {
9 if(argc != 2)
10 goto out;
11 FILE *fp;
12 char *str[MAX_CHAR];
13 char buffer[MAX_CHAR];
14 char *revstr[MAX_CHAR];
15 int curr_line = 0, i = 0, j =0;
16
17 fp = fopen(argv[1],"r");
18
19 out:
20 if (fp == NULL && argc != 2){
21 printf("File cannot be found or read failed.\n");
22 return -1;
23 }
24
25 /* printf("This part of the program reverses the input text file, up to a maximum of 100 lines and 80 characters per line.\n The reversed file, from the last (or 100th) line to the first, is the following:\n\n"); */
26
27 /* fgets reads one line at a time, until 80 chars, EOL or EOF */
28 while(curr_line < MAX_LINES && ((fgets(buffer, MAX_CHAR, fp)) != NULL)){
29
30 str[i] = buffer;
31 ++j;
32 ++i;
33 }
34
35 for(i = 0; i < 4; ++i)
36 printf("%s \n", str[i]);
37
38
39 printf("END OF PROGRAM RUN.");
40 return 0;
41 }
在同一目录中,我有一个包含以下行的“txt”文件:
is this a test
this is a test
this is not a test
当我编译并运行程序时 (./a.out txt),我得到以下输出:
this is not a test
this is not a test
this is not a test
END OF PROGRAM RUN.
显然这意味着它正在覆盖相同的位置,但我不确定如何纠正这个问题(如前所述,指针对我来说仍然很陌生)。谁能澄清这里发生了什么?我需要改用二维数组吗?任何帮助将不胜感激。
最佳答案
buffer
问题是你一直在str[i]
中存储指针buffer
,但是buffer
只包含读取的最后一行。
您必须复制读取的行,可能使用 strdup()
:
str[i++] = strdup(buffer);
需要担心错误检查,但从概念上讲,这是处理错误的一种简单方法。
Could you please clarify as to why that is? The declaration of
fgets()
beingchar *fgets()
, I got the impression that it momentarily returns a pointer to the line currently read from the file!
实际上,fgets()
返回两个值之一:
buffer
,指向传递给函数的字符串的指针。特别是,它不会自动为您创建新的存储空间。如果你想要那个,你可能需要看看 readline()
来自 POSIX。
因此,您对 fgets()
的调用每次都使用相同的数组 buffer
,因此读取的每个后续行都会覆盖前一行。这就是为什么您需要以某种方式复制每一行,以及为什么我建议使用 strdup()
作为一种简单的方法来做到这一点。
strdup()
函数是 POSIX 的一部分,但不是标准 C。但是,它很容易实现:
char *strdup(const char *str)
{
size_t len = strlen(str) + 1;
char *dup = malloc(len);
if (dup != 0)
memmove(dup, str, len);
return(dup);
}
您应该尽可能避免使用 goto
,使用这段代码很容易做到这一点。您的双重用途错误消息实际上最终会使省略参数或提供过多参数的用户感到困惑。提供两条单独的错误消息,一条是关于程序的参数数量不正确,另一条是关于未能打开文件。
if (argc != 2)
{
fprintf(stderr, "Usage: %s file\n", argv[0]);
exit(1);
}
fp = fopen(argv[1], "r");
if (fp == 0)
{
fprintf(stderr, "%s: failed to open file %s (%d: %s)\n",
argv[0], argv[1], errno, strerror(errno));
exit(1);
}
注意应该在stderr
上报告错误,而不是stdout
;这就是错误输出 channel 的用途。
关于c - 使用 fgets 将行复制到字符串数组中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14452335/
我有以下代码:(cpp14) static int const max_len = 1000; FILE* m_in_log = NULL; FILE* m_log = NULL; ... ... b
标准库函数 fgets() 有两个缺点: 函数的第二个参数是int类型 它在提供的缓冲区中留下一个尾随的换行符 我做了一个类似于 fgets() 的简单函数,排除了上述缺点,试图提高我的一个程序的效率
我看过各种关于在 fgets 之前使用 scanf 可能会导致问题的帖子。但是,对我来说,为什么在 AFTER 之后添加它会导致第一个语句中断,这是没有意义的。 这就是我所拥有的: unsigne
这可能是一个非常愚蠢的问题(可能),但我正在使用 fgets 处理文本文件。没什么太花哨的: while (fgets(buf, sizeof(buf), inputFile) != NULL) 正如
如果我想使用 fgets 从文本中读取多行,根据我的教科书,我会这样做: char str[53]; ... while(fgets(str, max, f)!=NULL){ ... }
我试图在完成未定义数量的另一个输入后获取用户输入。但问题是 while 循环之后的第二个 fgets 永远不会被调用。我用 EOF 结束循环,也许这就是错误。但我不知道应该如何结束循环。 另一个有趣的
我正在尝试将文件中的邮政编码读入 Object * 数组。文件包括123 无处不在柯克兰加州99223 我的 .h 文件看起来像 typedef struct { char *street;
我已经实现了我自己的 fgets(即 myfgets)。当我的文件中有必须由 myfgets 函数读取的 NULL 字符串时,它会打印所有字符串(好)但有一些垃圾(坏),但如果我使用预定义 fgets
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: Why does printf not flush after the call unless a newl
假设 FILE* 有效,请考虑: char buf[128]; if(fgets(buf,sizeof buf,myFile) != NULL) { strlen(buf) == 0; //ca
void verification() { char pass[50]; printf(" Enter Password : "); fgets(pass, 50, stdin
我想知道如何从汇编代码中调用fgets。我读到的这些问题与这个问题完全相同:How to call fgets in x86 assembly?这个:How to use c library func
我有一个程序,用户可以在其中输入两个输入。由于我无法控制用户输入的内容,因此用户可以超过数组的固定大小。由于 fgets() appends 在空字符之前保留了一个换行符,如果用户超出预期大小时换行符
我想使用 fgets() 简单地读取一行并去掉换行符。这是否适用于所有情况,包括 Windows 和 UN*X? fgets(buf, sizeof buf, stdin); strtok(buf,
我正在运行以下代码: #include #include #include int main(){ FILE *fp; if((fp=fopen("test.txt","r"))==N
如果我有以下缓冲区: char buffy[40]; // includes \0 at the end fgets 函数应该有 STLEN 40 还是 39?为什么? char buffy[40];
使用fgets输入字符串,我对读取的字符串长度有疑问。 例如,考虑以下程序。 char str[50]; int i; int len; printf("Enter name:\n"); fgets(
fgets()函数是否将文件指针自动移动到该位置,直到我提到的大小参数为止? 例如 : p.txt文件的内容是“我是个好男孩”。使用fgets(a,5,fp1)之后 文件指针会向前移动5个位置吗? 在
gcc 4.4.4 c89 我正在读取一个文件,在该文件中,我必须对每一行进行一些操作。 在一个循环中,我调用了 2 个函数,它们传入此文件指针以对文件执行 2 种不同的操作。 但是,当我使用 fge
我试图从文件中获取我的 9x9 数字网格并将其存储在 2D int 数组中。为此,我编写了代码,以便它使用文件中的 fgets(),然后将 char 数组值转换为 int,然后将其存储在数组中的相应位
我是一名优秀的程序员,十分优秀!