如何将 std::istream 转换为 std::wistream



我有一个 istream,有些代码需要一个 wistream。

我真的希望源流零的每个字符都扩展到wchar_t。 我不关心代码页,我不关心本地化,我只想无缝地通过管道传输这个输入,最快的方法是什么?

您可以编写读取缓冲区以使istream中的底层流缓冲区适应您的wistream

class adapting_wistreambuf : public wstreambuf {
  streambuf *parent;
public:
  adapting_istreambuf(streambuf *parent_) : parent(parent_) { }
  int_type underflow() {
    return (int_type)parent->snextc();
  }
};

后:

istream &somestream = ...;
adapting_wistreambuf sb(somestream.rdbuf());
wistream wistream(&sb);
// now work with wistream

这只实现了 streambof 接口的最低限度。如果需要,可以添加缓冲区管理代码以提高性能。

我认为如果不在std::wistream周围编写复杂的*包装器来提供std::istream接口,这是不可能的。两者之间的区别在于从模板实例化这些类时使用的 char 类型,从而使std::istream (char_type = char ( 和 std::wistream (char_type = wchar_t ( 在两个不同的类层次结构中。它们仅将 std::ios_base 共享为公共基类,这不会为您当前的问题提供有用的东西。

因此,以下代码段将无法编译(它尝试将 s2 的内部std::streambuf替换为 s1 的内部,从而使 s2 上的 I/O 操作对文件中的数据执行(:

std::wifstream     s1 ;
std::istringstream s2 ;
// assuming s1 and s2 are correctly initialized ...
s2.ios::rdbuf(s1.rdbuf()) ;  // will not compile, unfortunately, unless the char type
                             // is the same for the two streams :(

我不知道一个简单的解决方案,也许可以使用Boost IOStreams过滤器(我从未尝试过:((。

如果您不处理二进制数据,并且流长度不是太大,请尝试将所有数据读取为字符串,然后将(如您所说为零扩展(转换为std::wistringstream

* : 复杂意味着您对 std::streams 有合理的了解。我承认我对溪流不舒服。

相关内容

  • 没有找到相关文章