将文本文件中的小写字符转换为大写字母,反之亦然

标签 c file io

我有一个包含以下文本的文本文件,没有换行符...

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/

相关文章:

C - 结构体定义

scala - 如何使用 Play 在 Scala 中读取文件! 2.0?

javascript - 如何在nodejs中的多个文件之间共享变量

Python - 替换多个文件中的数字

c - Posix 调用 'read' 不读取整个文件

haskell - Haskell 内部的 IO 实现

c++ - 如何将 char *(字符指针)转换为 PCSZ?

c - 如何使用 Perl/其他东西搜索和替换 C 源代码中的特定代码行?

c - 通过递归取余

multithreading - 在Linux上通过SSH运行多线程Perl脚本的问题