我有一个包含以下文本的文本文件,没有换行符...
Hello World
我想将小写字符转换为大写字符,反之亦然,这样同一个文本文件将以以下文本结束......
hELLOW wORLD
不幸的是,当我运行我的代码时,它进入了无限循环。当我单步执行代码时,我看到 fseek() 正如预期的那样在第一个循环中返回一个字节,但在第二个和后续循环中返回两个字节。我不明白为什么它返回两个字节而不是一个字节。为什么会这样?有人可以帮忙吗?
这是我的代码...
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
int main()
{
FILE *fp;
int ch;
long offset;
fp = fopen("c:\\users\\domenic\\desktop\\test.txt", "r+");
if (fp == NULL)
{
printf("error: unable to open file\n");
exit(1);
}
offset = ftell(fp);
while (1)
{
ch = fgetc(fp);
if (ch == EOF)
break;
if (isupper(ch))
{
fseek(fp, offset, SEEK_SET);
fputc(tolower(ch), fp);
}
else if (islower(ch))
{
fseek(fp, offset, SEEK_SET);
fputc(toupper(ch), fp);
}
offset = ftell(fp);
}
fclose(fp);
return 0;
}
最佳答案
如果我理解你只是想在整个文件中将 upper 更改为 lower 并将 lower 更改为 upper ,你可能会让自己变得比需要的更难。
在我们研究使事情变得更简单的方法之前,让我们谈谈如何在代码中避免魔数(Magic Number) 和硬编码路径。 C 为 main
提供了一个定义,它允许您为代码提供参数以避免对值(例如文件/路径名)进行硬编码——使用它们。带参数的 main
的正确调用是:
int main (int argc, char *argv[])
(或者您将看到等效的 int main (int argc, char **argv)
)
不带参数的调用是int main (void)
。
现在开始讨论手头的问题。正如我在评论中提到的那样,在处理 ASCII 时,控制大小写的 bit 是第 6 位——根据讨论,如果您正在处理 EBCDIC,则 *case-bit 是第 7 位。正如@chux 所指出的,两者都可以通过为两者确定适当的位
A ^ a来无缝处理(结果是
32,例如
(1 << 5 )用于 ASCII,
64或
(1 << 6)用于 EBCDIC。要打开/关闭任何位,您只需将 *case-bit* 与当前字符
(A-Za-z)。到目前为止,任何字符
'c',您希望切换大小写,您只需将它与
异或A ^ a`,例如
if (('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z'))
c ^= A ^ a;
如果 c
是大写字母,则现在是小写字母,反之亦然。
要对整个文件执行此操作,将要转换的文件名作为程序的第一个参数(如果没有给出参数,则默认从 stdin
读取)并输出大小写转换后的结果到 stdout
,您可以执行如下简单的操作:
#include <stdio.h>
int main (int argc, char **argv) {
int c;
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 ((c = fgetc(fp)) != EOF) /* read each char */
/* is it a letter ? */
if (('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z'))
putchar (c ^ ('A' ^ 'a')); /* toggle case */
else
putchar (c); /* just output */
if (fp != stdin) fclose (fp); /* close file if not stdin */
return 0;
}
示例输入
$ cat dat/captnjack.txt
This is a tale
Of Captain Jack Sparrow
A Pirate So Brave
On the Seven Seas.
示例使用/输出
$ ./bin/case_toggle < dat/captnjack.txt
tHIS IS A TALE
oF cAPTAIN jACK sPARROW
a pIRATE sO bRAVE
oN THE sEVEN sEAS.
如果要将输出写入新文件,只需重定向输出,例如
$ ./bin/case_toggle < dat/captnjack.txt > dat/captnjack_toggled.txt
这会将大小写切换的输出写入 dat/captnjack_toggled.txt
。
检查一下,如果您还有其他问题,请告诉我。
关于将文本文件中的小写字符转换为大写字母,反之亦然,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46715132/