对string类型或含string的自定义数据类型,进行读写二进制文件时引发的异常: 写入位置时发生访问冲突的分析

 下面代码是想将一个含有string类型的struct类型,写入二进制文件中,并将其从二进制文件中读取出来输出

#include<iostream>
#include<fstream>
#include<string>
using namespace std;
struct student
{
	int number;
	string name;
};

int main()
{
	student stu;
	cin >> stu.number >> stu.name;

	ofstream ofile("CString.dat", ios::out | ios::binary);;	
	ofile.write((char*)(&stu), sizeof(stu));
	ofile.close();

	ifstream ifile("CString.dat", ios::in | ios::binary);
	student stu1;
	ifile.read((char*)(&stu1), sizeof(stu1));
	ifile.close();

	cout << stu1.number << endl << stu1.name << endl;

	return 0;
}

但上述代码会触发断点:

0x00D28C4B 处(位于 try.exe 中)引发的异常: 0xC0000005: 写入位置 0xDDDDDDDD 时发生访问冲突。

触发断点的位置:使用read函数,读取二进制文件时

 以下为我的看法:

二进制文件是以字节为单位进行读写文件的。而对于string类型,其所占的字节数为28个。

对于二进制文件的读写函数中均有sizeof(需要读或写入的数据类型)作为参数

当将stu的数据输出到二进制文件中,是正常的(比如编译器知道碰到'\0'时,该string类型的数据就输出完毕了)

但是到了读取数据进入stu1中时就出问题了!!!

ifstream ifile("CString.dat", ios::in | ios::binary);
student stu1;
ifile.read((char*)(&stu1), sizeof(stu1));
ifile.close();

因为这里传进去ifile.read的参数为sizeof(stu1),sizeof(stu1)值固定为32个字节(整型4+字符串类型28),而实际上string类型中存储数据的字节数有可能大于28也有可能小于28个字节(string类通过char类型指针实现)

读取时以字节为单位,那么编译器就不知道应该在该二进制文件中读取多少个字节的数据放到stu1中(这里有可能是读取了32个字节的数据,但实际上写入的数据并没有这么大,导致读指针读取到了没有写入数据的位置,导致读写异常)

实际上在二进制文件中,写入通过指针指向的数据(write函数中参数写sizeof(该指针) 时),在读取数据时都会触发该异常。

解决方法:

将结构体中的string类型改为一个恰当大小的字符数组


假如是对文本文件进行上述操作呢

#include<iostream>
#include<fstream>
#include<string>
using namespace std;
struct student
{
	int number;
	string name;
};
int main()
{
	ofstream ofile("CString.txt", ios::out);
	student stu;
	cin >> stu.number >> stu.name;
	ofile << stu.number << stu.name;
	cin >> stu.number >> stu.name;
	ofile << stu.number << stu.name;
	ofile.close();

	ifstream ifile("CString.txt", ios::in);
	student stu1, stu2;
	stu2.name = "没有数据";
	stu2.number = 0;
	ifile >> stu1.number >> stu1.name;
	ifile >> stu2.number >> stu2.name;
	ifile.close();
	cout << stu1.number << endl << stu1.name << endl;
	cout << stu2.number << endl << stu2.name << endl;
	return 0;
}

答案是:不会引发异常

因为对于文本文件,在写入文件时是将每个字符先转换为ASCII码(每个ASCII码占一个字节),再写入文件中。

但是在读取文件时,对于读入一个string类型数据,会将从其开始读取string类型数据位置开始(上一个数据读取完的末尾),一直读取到文件末尾。其余的数据类型则无法读到数据。(如上述的stu2的name,number)

 总的来说:

在进行写文件和读文件时就尽量不要使用string类型数据,尽量使用一个合适大小的字符数组


个人看法和理解,如果有误,望指出啊!!!

如果有其他看法,欢迎一同交流啊!!!

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是用字节缓冲流实现二进制文件读写操作的Java代码,可以将wn.mp3文件复制为hl.mp3文件: ```java import java.io.*; public class BinaryFileCopy { public static void main(String[] args) { String sourceFileName = "wn.mp3"; // 原始文件String destinationFileName = "hl.mp3"; // 目标文件名 int bufferSize = 1024 * 1024; // 缓冲区大小 try (InputStream inputStream = new BufferedInputStream(new FileInputStream(sourceFileName), bufferSize); OutputStream outputStream = new BufferedOutputStream(new FileOutputStream(destinationFileName), bufferSize)) { byte[] buffer = new byte[bufferSize]; int len; while ((len = inputStream.read(buffer)) != -1) { outputStream.write(buffer, 0, len); } outputStream.flush(); System.out.println("文件复制成功!"); } catch (IOException e) { e.printStackTrace(); } } } ``` 上述代码中,我们使用了字节缓冲输入流BufferedInputStream和字节缓冲输出流BufferedOutputStream,这两个流都是装饰器流,它们可以在内部维护一个缓冲区,可以提高读写效率。在使用这两个流,我们需要提供一个缓冲区大小,这里我们设置为1MB。 在读写文件,我们使用了一个byte数组作为缓存,每次从输入流中读取数据,然后写入到输出流中,直到读取完毕。最后,我们需要调用输出流的flush()方法,将缓存中的数据刷入文件中,确保文件的完整性。 注意:这里我们使用了Java 7中引入的try-with-resources语句,它可以自动关闭流,无需手动关闭。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值