读取和写入同一文件fstream

时间:2016-09-11 13:01:50

标签: c++ stream ifstream ofstream

我想更新现有的json文件。

这是示例json文件:

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

来自程序的日志:

Operation successfully performed
100
"Test1"
51.32
46.32
Done

看起来每个方面都按预期工作......

如果我将fstream更改为ifstream以进行读取,稍后将其更改为将其写入工作...

我尝试使用调试器,因为我看到我在basic_ostream对象中有错误的数据...但我不知道为什么,我使用来自字符串的数据并更正(更新数据)。 知道什么是错的:-)?

1 个答案:

答案 0 :(得分: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;
}