jce*_*ed2 17 c++ file seek language-lawyer c++20
C++ 使用该streamoff
类型来表示(文件)流中的偏移量,并在 [stream.types] 中定义如下:
using streamoff = implementation-defined ;
类型 streamoff 是有符号基本整数类型之一的同义词,其大小足以表示操作系统的最大可能文件大小。287)
287) 通常很长很长。
这是有道理的,因为它允许在大文件中查找(而不是使用long
可能只有 32 位宽的 )。
[filebuf.virtuals] 定义了basic_filebuf
在文件中查找的函数,如下所示:
pos_type seekoff(off_type off, ios_base::seekdir way, ios_base::openmode which = ios_base::in | ios_base::out) override;
off_type
等价于streamoff
,参见 [iostreams.limits.pos]。然而,标准接着解释了函数的效果。我被最后一句话激怒了,它需要调用fseek
:
效果:让
width
表示a_codecvt.encoding()
。如果is_open() == false
, 或off != 0 && width <= 0
,则定位操作失败。否则,如果way != basic_ios::cur
或off != 0
,并且如果最后一个操作是输出,则更新输出序列并写入任何非移位序列。接下来,寻找新的位置:如果width > 0
,调用fseek(file, width * off, whence)
,否则调用fseek(file, 0, whence)
。
fseek
接受一个long
参数。如果off_type
和streamoff
被定义为long long
(如标准所建议的那样),这可能会导致long
调用时向下转换fseek(file, width * off, whence)
(导致可能难以诊断错误)。这streamoff
首先对引入该类型的整个基本原理提出了质疑。
这是故意的还是标准中的缺陷?
我认为您从中得出的结论是,C++ 流之间存在不匹配并且fseek
会导致运行时错误,这是不正确的。情况似乎是:
在long
64 位的系统上,streamoff
定义为long
,seekoff
函数调用fseek
.
在long
32 位但操作系统支持 64 位文件偏移量的系统上,streamoff
被定义为long long
并seekoff
调用一个函数,该函数被称为fseeko
或fseeko64
接受 64 位偏移量。
这是seekoff
我的 Linux 系统上定义的片段:
#ifdef _GLIBCXX_USE_LFS
if (!fseeko64(_M_file, __off, __whence))
__ret = std::streampos(ftello64(_M_file));
#else
if (!fseek(_M_file, __off, __whence))
__ret = std::streampos(std::ftell(_M_file));
#endif
Run Code Online (Sandbox Code Playgroud)
LFS 代表大文件支持。
结论:虽然标准提出了一个streamoff
表面上与seekoff
invoke要求冲突的定义fseek
,但库设计者明白他们必须调用fseek
接受操作系统支持的全部偏移量的变体。
归档时间: |
|
查看次数: |
595 次 |
最近记录: |