我需要有关 s 的不可复制性质的帮助[io](f)stream
。
我需要在fstream
s 周围提供一个 hackish 包装器,以便在 Windows 上处理文件名中包含 unicode 字符的文件。为此,我设计了一个包装函数:
bool open_ifstream( istream &stream, const string &filename )
{
#ifdef __GLIBCXX__
FILE* result = _wfopen( convert_to_utf16(filename).c_str(), L"r" );
if( result == 0 )
return false;
__gnu_cxx::stdio_filebuf<char>* buffer = new __gnu_cxx::stdio_filebuf<char>( result, std::ios_base::in, 1 );
istream stream2(buffer);
std::swap(stream, stream2);
#elif defined(_MSC_VER)
stream.open( convert_to_utf16(filename) );
#endif
return !!stream;
}
当然,这std::swap
条线是罪魁祸首。我也尝试从函数返回流,但它会导致同样的问题。a 的复制构造函数std::istream
是delete
d。我也尝试过std::move
,但没有帮助。我该如何解决这个问题?
编辑:Keep It Simple (TM)
感谢@tibur 的想法,我终于找到了一个很好的方法,但又很实用。从某种意义上说,它仍然是骇人听闻的,因为它取决于所使用的 Windows 标准 C++ 库,但由于只有两个真正的库在使用,所以对我来说这不是问题。
#include <fstream>
#include <memory>
#if _WIN32
# if __GLIBCXX__
# include<ext/stdio_filebuf.h>
unique_ptr<istream> open_ifstream( const string &filename )
{
FILE* c_file = _wfopen( convert_to_utf16(filename).c_str(), L"r" );
__gnu_cxx::stdio_filebuf<char>* buffer = new __gnu_cxx::stdio_filebuf<char>( c_file, std::ios_base::in, 1 );
return std::unique_ptr<istream>( new istream(buffer) );
}
# elif _MSC_VER
unique_ptr<ifstream> open_ifstream( const string &filename )
{
return unique_ptr<ifstream>(new ifstream( convert_to_utf16(filename)) );
}
# else
# error unknown fstream implementation
# endif
#else
unique_ptr<ifstream> open_ifstream( const string &filename )
{
return unique_ptr<ifstream>(new ifstream(filename) );
}
#endif
在用户代码中:
auto stream_ptr( open_ifstream(filename) );
auto &stream = *stream_ptr;
if( !stream )
return emit_error( "Unable to open nectar file: " + filename );
这取决于 C++0x<memory>
和auto
关键字。当然,您不能只close
使用结果stream
变量,但 GNU Libstdc++std::istream
析构函数确实负责关闭文件,因此在任何地方都不需要额外的内存管理。