我发现在VS2010中,当打开正好4294967295字节的文件时,seekg函数不能正常工作。
我正在使用简单的代码:
#include <iostream>
#include <fstream>
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{
std::ifstream file;
// cmd: fsutil file createnew tmp.txt 4294967295
file.open(L"c:/tmp.txt", ifstream::in | ifstream::binary);
if(!file.is_open())
return -1;
file.seekg(0, std::ios::end);
auto state = file.rdstate();
// this condition shoots only when size of the file is equal to 4294967295
if((state & ifstream::failbit)==ifstream::failbit)
{
std::cout << "seekg failed";
}
// after seekg failed, tellg returns 0
std::streampos endPos = file.tellg();
return 0;
}
与 4294967294 和 4294967296 文件相同的代码可以正常工作。
有人知道这个问题的解决方案吗?
更新:
看起来问题出在这里:
template<class _Statetype>
class fpos
{
__CLR_OR_THIS_CALL operator streamoff() const
{ // return offset
return ((streamoff)(_Myoff + _FPOSOFF(_Fpos)));
}
}
正好在
_FPOSOFF(_Fpos)
在哪里
#define _FPOSOFF(fp) ((long)(fp))
所以它需要 4294967295 并将其转换为 -1 !
换句话说,这样的代码会失败
//returns -1, even if sizeof(fpos_t)=8
fpos_t pos = _FPOSOFF(4294967295);
_Myoff、_Fpos、streamoffset 为 64 位
如果所有类型都是 64 位,他们为什么要进行这种转换!?我不知道 ))