读写同一文件的 fstream

4

我希望更新现有的 JSON 文件。

这是一个示例 JSON 文件:

{
   "Foo": 51.32,
   "Number": 100,
   "Test": "Test1"
}

程序日志:

从程序中获取的日志:

Operation successfully performed
100
"Test1"
51.32
46.32
Done

看起来一切都按预期工作...

如果我将 fstream 更改为 ifstream 以进行读取,然后再将 ofstream 更改为写入,它可以正常工作...

我尝试使用调试器并且看到 basic_ostream 对象中有错误的数据...但是我不知道为什么,我使用从已更正(更新)的字符串数据中获取的数据。

有什么想法吗 :-)?


解决这类问题的正确工具是调试器。在向Stack Overflow提问之前,您应该逐行查看代码。如需更多帮助,请阅读如何调试小程序(Eric Lippert撰写)。至少,您应该[编辑]您的问题,包括一个最小、完整和可验证的示例,以重现您在调试器中观察到的问题。 - πάντα ῥεῖ
1
在尝试重写文件内容之前,您需要使用 clear() 和很可能的 seek()。请注意,如果新内容比现有内容短,这不会截断文件的大小,您最终会得到末尾的垃圾数据。 - Sam Varshavchik
1
帖子已更新 :-) - user3490777
1个回答

1
你有几个问题。
首先,命令json json_data(fs);读取到文件末尾并设置了EOF标志。流将停止工作,直到该标志被清除为止。
其次,文件指针位于文件末尾。如果你想要覆盖该文件,你需要将指针再次移回开头:
if (fs.is_open())
{
    json json_data(fs); // reads to end of file
    fs.clear(); // clear flag
    fs.seekg(0); // move to beginning

很不幸,这仍然不能完全解决问题,因为如果你写回的文件比读入的文件小,那么一些旧数据会标记在新数据的末尾:
    std::cout << "Operation successfully performed\n";
    std::cout << json_data.at("Number") << std::endl;
    std::cout << json_data.at("Test") << std::endl;
    std::cout << json_data.at("Foo") << std::endl;

    json_data.at("Foo") = 4.32; // what if new data is smaller?

Json文件:

{
   "Foo": 4.32, // this number is smaller than before
   "Number": 100,
   "Test": "Test1"
}} // whoops trailing character from previous data!!

在这种情况下,我会简单地打开一个文件进行读取,然后再打开另一个文件进行写入,这样更不容易出错,并表达了覆盖一切的意图。
类似这样:
#include "json.hpp"
#include <iostream>
#include <fstream>
#include <string>

using json = nlohmann::json;

void readAndWriteDataToFile(std::string fileName) {

    json json_data;

    // restrict scope of file object (auto-closing raii)
    if(auto fs = std::ifstream(fileName))
    {
        json_data = json::parse(fs);

        std::cout << "Operation successfully performed\n";
        std::cout << json_data.at("Number") << std::endl;
        std::cout << json_data.at("Test") << std::endl;
        std::cout << json_data.at("Foo") << std::endl;
    }
    else
    {
        throw std::runtime_error(std::strerror(errno));
    }

    json_data.at("Foo") = 4.32;
    std::cout << json_data.at("Foo") << std::endl;
    std::string json_content = json_data.dump(3);

    if(auto fs = std::ofstream(fileName))
    {
        fs.write(json_content.data(), json_content.size());
        std::cout << "Done" << std::endl;
    }
    else
    {
        throw std::runtime_error(std::strerror(errno));
    }

}

int main()
{
    try
    {
        std::string fileName = "C:/new/json1.json";
        readAndWriteDataToFile(fileName);
    }
    catch(std::exception const& e)
    {
        std::cerr << e.what() << '\n';
        return EXIT_FAILURE;
    }
    return EXIT_SUCCESS;
}

您IP地址为143.198.54.68,由于运营成本限制,当前对于免费用户的使用频率限制为每个IP每72小时10次对话,如需解除限制,请点击左下角设置图标按钮(手机用户先点击左上角菜单按钮)。 - user3490777

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接