可变参数模板运算符<<
Variadic template operator<<
我正试图将我的一些函数foo()
更改为operator<<()
,只是为了让一些"半C/半C++"代码看起来更像C++。然而,发生的事情是,我陷入了以下转换步骤:
template <class... T>
inline const size_t foo(const T&... data) {
return sizeof...(T);
}
struct bar {
template <class... T>
inline const size_t operator<<(const T&... data) {
return sizeof...(T);
}
};
int main(int argc, char *argv[]) {
bar a;
std::cout << ">>> length " << foo(1, 2, 3) << std::endl;
std::cout << ">>> length " << (a << 1 << 2) << std::endl;
std::cout << ">>> length " << (a << 1 << 2 << 3) << std::endl;
std::cout << ">>> length " << (a << 1 << 2 << 3 << 4) << std::endl;
}
来自输出:
$ ./a.out
>>> length 3
>>> length 4
>>> length 32
>>> length 512
我得出的结论是,第一次计算是在a << 1
上进行的,随后的值会相应地偏移。然而,我不知道如何重写foo()
,以便为struct bar
的用户提供operator<<()
接口——当然,在不更改foo()
语义的情况下。
如果无法将class T...
作为参数传递给operator<<()
,那么该函数的效率自然会低于foo()
,因为它会被调用多次。对此,是否有任何合理的C++构造,或者坚持使用foo()
是唯一/最好的选择?
上下文:
这些foo()
功能是网络通信的发送器/接收器。我认为最好提供一个更"C++"的接口,带有发送方/接收方流,使用<<
和>>
运算符可写/可读,而不是使用常规函数foo(...)
。
该语言正在执行您要求它执行的操作。
对于关联性,以下是等价的(<<
和>>
是从左到右关联的):
a << 1 << 2
(a << 1) << 2
调用a << 1
调用您的用户定义运算符,然后返回一个size_t
。这就是为什么下一个调用的类型如下:size_t << int
(这是一个简单的逐位移位)。
您需要使用表达式模板。其想法如下(此处为示例):
template<typename... args>
struct stream_op
{
};
template<typename... A, typename B>
stream_op<A..., B> operator<<(stream_op<A...> a, B b)
{
// Do stuff
}
因此,出现以下情况(将a
作为stream_op<>
):
a << 1 << 2
------
|
v
-------------------------------------------
stream_op<int> operator<<(stream_op<>, int) << 2
-------------- ---
| |
| +---------------------------+
v v
-------------- ---
stream_op<int> << int
-------------- ---
| |
| +---------------------------+
+----------------------------+ |
v v
-------------- ---
stream_op<int,int> operator<<(stream_op<int>, int)
------------------
|
v
------------------
stream_op<int,int> // <- Expected result
然后,您只需要放置一个方法来将stream_op
转换为int(或任何您想要的)。
关于性能的注意事项:使用这些表达式模板,一部分数据被编码在类型中,因此通常它应该像直接调用foo(...)
一样快。
相关文章:
- 如何防止clang格式在流运算符调用之间添加换行符<<
- 如何显式调用运算符<<
- 为什么COUT在朋友函数中不起作用,该功能超载了操作员&lt;&lt;这是一个iStream运算符
- C++运算符<<调用::ostream而不是std::osttream
- BOOST ::变体无法解决运算符&lt;&lt;对于STD :: Ostream
- 过载输出<<用于类的运算符,以打印其中的元组
- C++ostream:没有运算符匹配<<&应在'&'代币
- 重载运算符<<:此运算符函数的参数太多
- C++继承运算符<<
- 重载运算符<<用于模板类.即使使用好友关键字也无法获得私人会员
- 如何过载<<用于YAML::Emitter的运算符,以序列化包含另一个自定义类的向量的自定义类
- 为什么字符串流运算符<<擦除原始值
- 关于使用运算符<<为新手提供C++中的模板
- 我已经完成了<<运算符重载,但它'It’不起作用
- 重载运算符<<输出地址而不是数据成员
- 错误:没有匹配'运算符<<"在'std::cout
- 重载运算符<<用于ostream语法
- 当运算符<存在时,为什么要定义 LT?
- log4cxx访问异常,使用<<运算符和宏
- 重载<<运算符错误C2804:二进制'运算符<<'参数太多