插入对流的右值引用是否合理有效
Is it reasonably efficient to insert into an rvalue reference to a stream?
我创建了一个自定义流类型,称之为error_stream
,它源自std::ostringstream
。我还为流制作了一个名为throw_cpp_class
的自定义操纵器(throw_cpp
是throw_cpp_class
的一个实例)。我的目标是使用以下语法:
error_stream s;
s << "some error " << message() << throw_cpp; // throw_cpp throws an exception containing contents of the stream.
我发现,通过定义一个插入运算符,将流的右值引用作为第一个操作数,我现在可以做到这一点:
error_stream() << "some error " << message() << throw_cpp;
插入运算符如下所示:
error_stream& operator<<(error_stream&& s, const throw_cpp_class&)
{
throw s.str();
return s;
}
这是怎么回事?为什么我可以返回类型为error_stream&&
的值,其中需要error_stream&
?(这会调用move构造函数吗?)。这是不是效率低下得可怕?(并不是说我真的在乎,因为这种例外应该很少见)。
使用此代码:
error_stream& operator<<(error_stream&& s, const throw_cpp_class&)
{
throw s.str();
return s;
}
您可以将error_stream&& s
作为error_stream&
返回,因为s
是左值,而不是右值。
"什么?"你问道?"但我看到&&
就在那里!"。C++的这一部分很棘手。当您看到type&& s
(而type
不是模板)时,这意味着变量是一个右值引用,它是一个"从"右值构建的引用。但它有一个名字:s
。所有有名字的东西都是左值。这就是为什么有时必须调用std::move
,因为您必须让编译器知道您希望它再次将该变量视为右值。
这会调用move构造函数吗?)。
不,它只是返回一个对左值s
的引用。
这是不是效率低下得可怕?(并不是说我真的在乎,因为这种例外应该很少见)。
没有,因为没有复制,甚至没有移动。
与您的实际问题无关,流的大多数过载是:
ostream& operator<<(ostream&& s, const T&)
那么这意味着,除非throw_cpp
是流式传输的第一个东西,否则将不会调用您的重载,因为前一个流式传输将返回ostream&
,而不是error_stream&&
。(请注意,它们应该是模板,但许多不是,这与要点无关)您必须将其转换回error_stream
。
此外,操纵器也不是这样工作的。操纵器是函数,当你将这些函数流式传输到流时,流会调用函数并将本身作为参数传递,所以你想要更像这样的东西:
template <class exception, class charT, class traits>
std::basic_ostream<charT,traits>& throw_cpp(std::basic_ostream<charT,traits>& os)
{
error_stream& self = dynamic_cast<error_stream&>(os); //maybe throws std::bad_cast
throw exception(self.str());
return os; //redundant, but the compiler might not know that.
}
这里它正在工作(使用字符串流)
T&&
是一个右值引用,但其值类别是引用(左值)的类别,因此它可以存储在左值引用中。在这种情况下也不会调用move/copy构造函数,因为它是通过引用获取/返回的。
我一点也不认为这是低效的,而是对右值引用的常见和惯用用法。
- 如果变量名称不跟在 char* 后面,const char* 是否有效?
- 钳制迭代器是否有效
- 检查由括号、方括号和大括号组成的一组方括号是否有效?
- 在函数内创建的对象的范围 - 如果在函数外部存储和访问引用,它们是否有效?
- 模板签名解析为 void(void) 被 GCC 拒绝;这是否有效C++?
- 我如何知道作为参数的size_t在函数中是否有效?
- 我的运算符重载是否有效<<(流插入)左操作数不是 ostream
- C++ 返回指向函数内定义的静态数组的指针是否有效?
- 此递归模板类型定义是否有效C++?
- 将 C 函数转换为 C++ 以检查数字是否有效
- 函数参数的名称与调用函数时使用的变量相同是否有效?
- 如何检查输入是否有效?
- 如何检查用户的输入是否有效以及我正在寻找的数字?
- 在函数中按值传递 unordered_map/unordered_set 是否有效? C++
- 如何检查isupper(cstr)是否有效?
- 将 std::transform 与 std::back_inserter 一起使用是否有效?
- 只需要知道我在c ++中打印模式的方式是否有效,或者有另一种方法可以有效地做到这一点
- 如何检查字符串格式在读取C++文本文件时是否有效?
- 在 c++ 中将对象设置为等于同一类的构造函数是否有效?
- 创建加密安全密码.并验证它是否有效