读写同一个文件fstream
Reading and writing to the same file fstream
我想更新现有的 json 文件。
这是示例 json 文件:
{
"Foo": 51.32,
"Number": 100,
"Test": "Test1"
}
来自程序的日志:
Operation successfully performed
100
"Test1"
51.32
46.32
Done
看起来一切都按预期工作...
如果我将 fstream 更改为 ifstream 以进行读取,然后将 ofstream 更改为写入它正在工作...
我尝试使用调试器,正如我所见,我在 basic_ostream 对象中有错误的数据...但我不知道为什么,我使用来自已更正(更新的数据)的字符串中的数据。
知道出了什么问题 :-) ?
你这里有一些问题。
首先,命令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;
}
我想更新现有的 json 文件。
这是示例 json 文件:
{
"Foo": 51.32,
"Number": 100,
"Test": "Test1"
}
来自程序的日志:
Operation successfully performed
100
"Test1"
51.32
46.32
Done
看起来一切都按预期工作...
如果我将 fstream 更改为 ifstream 以进行读取,然后将 ofstream 更改为写入它正在工作...
我尝试使用调试器,正如我所见,我在 basic_ostream 对象中有错误的数据...但我不知道为什么,我使用来自已更正(更新的数据)的字符串中的数据。 知道出了什么问题 :-) ?
你这里有一些问题。
首先,命令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;
}