C++:搜索并写入文件以替换字符串

标签 c++ file io

所以我创建了一个简单的测试应用程序。它的目的是读取或写入文本文件。它正在寻找字符串“固件:”。这个想法是一个设备将有一个包传输到它并存储,然后激活。我想存储该包的名称(例如:Update1),当我需要时,查询它并取回包名称(从文件中读取)。

问题 1: 似乎没有写入文件。 问题 2: 当我让它写入文件时,如果存在“Firmware: pkgName”这一行,我该如何用新的包名称替换 pkgName?

这个程序运行并编译,所以随意将它扔进 IDE 或文本编辑器,它只是没有完全做我想要的。我使用 log.txt 作为我的示例文件来读取。我相信有一种更简单的方法可以做到这一点。我已经有一段时间了,我已经被难住了。

感谢您的帮助!

#include <cstdlib>
#include <iostream>
#include <string>
#include <fstream>

using namespace std;

int writeFile(string filename, string pkgName);
int readFile(string filename);

int main(void) {
    string filename = "log.txt", userInput, pkgName;
    system("touch log.txt");

    while (true) {
        cout << "Write or Read?\n>>>";
        cin >> userInput;

        if (userInput == "Write") {
            cout << "What's your package name?\n>>>";
            cin >> pkgName;
            writeFile(filename, pkgName);
        } else if (userInput == "Read") {
            readFile(filename);
        } else {
            cout << "Invalid Input";
        }
    }
}

int writeFile(string filename, string pkgName) {
    bool found, exists = true;
    string line, word;
    fstream firmwareFile;
    int pos, size, num = 1;
    firmwareFile.open(filename.c_str(), ios::in | ios::out | ios::app);
    //firmwareFile << "This is a test.";
    if (firmwareFile.is_open()) {
        while (!found) {
            getline(firmwareFile, line);
            pos = line.find("Firmware: ");
            if (pos != string::npos) {
                //Found It
                found = true;
                exists = true;
                size = line.size();
                word = line.substr(10, ((size - pos) + 10));
                cout << word << " is the old package name" << endl;
                //TODO - Replace that with the new package name
                firmwareFile.close();
            } else if (firmwareFile.eof()) {
                //Doesn't Exist
                found = true;
                exists = false;
                cout << "No last firmware package available.\n";
            } else {
                //Still Looking
                found = false;
                cout << "Searching line #" << num <<  endl;
            }
            num++;
            if (!exists) {
                firmwareFile << endl << "Firmware: " << pkgName << endl;
                cout << "Wrote package name to file\n" << endl;
            }
            firmwareFile.close();
        }   
    } else {
        cout << "Unable to open file";
    }
}

int readFile(string filename) {
    bool found;
    string line, word;
    fstream firmwareFile;
    int pos, size, num = 1;
    firmwareFile.open(filename.c_str(), ios::in | ios::out | ios::app);
    if (firmwareFile.is_open()) {
        while (!found) {
            getline(firmwareFile, line);
            pos = line.find("Firmware: ");
            if (pos != string::npos) {
                found = true;
                size = line.size();
                word = line.substr(10, ((size - pos) + 10));
                cout << word << endl;
            } else if (firmwareFile.eof()) {
                found = true;
                cout << "No last firmware package available.\n";
                firmwareFile << endl << "Firmware: ";
            } else {
                cout << "Searching line #" << num << endl;
            }
            num++;
        }
        firmwareFile.close();
    } else {
        cout << "Unable to open file";
        firmwareFile.close();
    }
    firmwareFile.close();
}

这是我的 log.txt 文件的内容(除了包含“固件:”的行之外的所有内容都是微不足道的)

You
Think
It's
Here
But
It's
Not
So
That
Sucks
Doesn't
It
?
Firmware: Update1
I
Hope
You
Caught
It
!

最佳答案

嗯。第一个问题,在您的函数写入文件中,您有一个变量 found,您在 while 条件 while (!found) { ... } 中对其进行测试。问题是当您第一次测试这个变量时,您没有将它设置为任何值。所以你在那里有未定义的行为。查看其余代码,我认为您应该使用 do while 循环 do { ... } while (!found); 而不是 while 循环。看起来你在 readFile 中也有同样的问题。

恐怕第二个问题是个坏消息。除非用长度完全相同的字符串替换,否则不能替换文件中的字符串。这只是文件在磁盘上组织方式的限制。实现替换的唯一方法是将整个文件读入内存,在内存中进行替换,然后将整个文件写回磁盘。

关于C++:搜索并写入文件以替换字符串,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11784735/

相关文章:

c++ - 间接链接库的符号可见性

C++ stod 函数等效

java - 如何拆分包含两个日期的文本行

c# - 创建缩略图并减小图像大小

Haskell sequencelistIO [a -> IO a] -> a -> IO a

c++ - Very Sleepy 分析器中花括号附近显示的时间测量值代表什么?

c++ - 这个复制构造函数是否创建了输入对象的相同且独立的深层拷贝?

bash - 用 bash 解析 ICS 文件

c++ - printf 到控制台窗口和文件?

java 使用扫描仪进行 I/O 和文件重定向