我需要将QFile
复制到另一个QFile
,因此我无法使用QFile::copy
。这是最原始的实现:
bool CFile::copyChunk(int64_t chunkSize, const QString &destFolder)
{
if (!_thisFile.isOpen())
{
// Initializing - opening files
_thisFile.setFileName(_absoluteFilePath);
if (!_thisFile.open(QFile::ReadOnly))
return false;
_destFile.setFileName(destFolder + _thisFileName);
if (!_destFile.open(QFile::WriteOnly))
return false;
}
if (chunkSize < (_thisFile.size() - _thisFile.pos()))
{
QByteArray data (chunkSize, 0);
_thisFile.read(data.data(), chunkSize);
return _destFile.write(data) == chunkSize;
}
}
这个片段并不清楚,但我只打算将一个二进制文件整体复制到另一个位置,只是在块中,这样我就可以为大文件提供进度回调和取消工具。
另一个想法是使用内存映射。我是不是该?如果是,那么我应该只映射源文件并仍使用_destFile.write
,还是应该映射两者并使用memcpy
?
我想这个问题与Qt没有关系,我认为答案应该是支持内存映射的任何文件I / O API的通用。
答案 0 :(得分:3)
好的,好吧,如果它必须是内存映射解决方案。这是一个:
QFile source("/tmp/bla1.bin");
source.open(QIODevice::ReadOnly);
QFile destination("/tmp/bla2.bin");
destination.open(QIODevice::ReadWrite);
destination.resize(source.size());
uchar *data = destination.map(0,destination.size());
if(!data){
qDebug() << "Cannot map";
exit(-1);
}
QByteArray buffer;
int chunksize = 200;
int var = 0;
do{
var = source.read((char *)(data), chunksize);
data += var;
}while(var > 0);
destination.unmap(data);
destination.close();
这仅将目标文件映射到内存中。我怀疑映射源文件也会有很大的不同。但这是具体测量的东西,而不是假设。
另一个问题是您是否可以一次将整个文件映射到内存中。不断地取消映射和重新映射肯定会降低性能。即使你使用Qt。诸如存储器映射之类的功能在不同平台上具有令人不安的不同行为的倾向,例如,映射到内存的最大文件大小可能不同。
答案 1 :(得分:0)
最佳方法是什么,总是在旁观者的眼中。这是至少一种工作较短的方法:
QFile source("/tmp/bla1.bin");
source.open(QIODevice::ReadOnly);
QFile destination("/tmp/bla2.bin");
destination.open(QIODevice::WriteOnly);
QByteArray buffer;
int chunksize = 200; // Whatever chunk size you like
while(!(buffer = source.read(chunksize)).isEmpty()){
destination.write(buffer);
}
destination.close();
source.close();
和记忆映射......我试图远离这样的事情。我永远不会确定他们是如何独立于平台。
答案 2 :(得分:-1)
使用此QFile :: map()方法:
QFile fs("Sourcefile.bin");
fs.open(QFile::ReadOnly);
QFile fd("Destinationfile.bin");
fd.open(QFile::WriteOnly);
fd.write((char*) fs.map(0, fs.size()), fs.size()); //Copies all data
fd.close();
fs.close();