#define print(msg) std::cout << msg << std::endl
#define print(msg) std::cout << msg << std::endl
我最近开始使用C++,我有一种强烈的冲动
#define print(msg) std::cout << msg << std::endl
这在所有情况下都能正确执行吗? 这是我所知道的唯一一种在味精中出现<<
时会起作用的公式(例如 "foo" << myInt
(。 也不
#define print(msg) std::cout << (msg) << std::endl // note: parens
也不是建议的答案
template<typename T>
void print(T const& msg) {
std::cout << msg << std::endl;
}
在这种情况下工作。 我也不关心用endl
刷新输出的效率与仅使用n
。
既然你提到你最近才开始使用C++,我想向你展示该语言提供的更好的替代方案:
template<typename T>
void print(T const& msg)
{
std::cout << msg << std::endl;
}
它接受任何类型的单个msg
参数,并通过std::cout
将其流式传输出去。
如注释中所述,std::endl
不仅插入新行,还会刷新流。这类似于printf
n
冲洗。如果你只是想要一个新行,你可能想要,最好明确地这样做:
std::cout << msg << 'n';
这是非常主观的,但是您编写一次代码并多次阅读。代码的其他维护者会想要理解你写了什么,所以只要写std::cout << msg << std::endl
你的意思。不要试图C++看起来像另一种语言。
只是为了好玩,这里有两个print
的可变参数C++11实现:一个在参数之间插入空格,另一个不插入。 (住在 ideone。
#include <iostream>
namespace with_spaces {
namespace detail {
std::ostream& print(std::ostream& os) {
return os;
}
template <typename T>
std::ostream& print(std::ostream& os, T&& t) {
return os << std::forward<T>(t);
}
template <typename T, typename U, typename... Args>
std::ostream& print(std::ostream& os, T&& t, U&& u, Args&&... args) {
return print(print(os, std::forward<T>(t)) << ' ', std::forward<U>(u), std::forward<Args>(args)...);
}
}
template <typename... Args>
void print(Args&&... args) {
detail::print(std::cout, std::forward<Args>(args)...) << std::endl;
}
}
namespace without {
namespace detail {
std::ostream& print(std::ostream& os) {
return os;
}
template <typename T>
std::ostream& print(std::ostream& os, T&& t) {
return os << std::forward<T>(t);
}
template <typename T, typename... Args>
std::ostream& print(std::ostream& os, T&& t, Args&&... args) {
return print(print(os, std::forward<T>(t)), std::forward<Args>(args)...);
}
}
template <typename... Args>
void print(Args&&... args) {
detail::print(std::cout, std::forward<Args>(args)...) << std::endl;
}
}
#include <iomanip>
int main() {
std::cout << std::boolalpha;
with_spaces::print(1, "foo", new int(3), 0xFFFFFFFFFFULL, 42, 0 == 1);
without::print(1, "foo", new int(3), 0xFFFFFFFFFFULL, 42, 0 == 1);
}
对我来说有趣的是,完成简单的单行宏可以执行的操作需要多少代码。
宏
被广泛使用,但不是很好的做法(尤其是在C++中(,因为它们可以隐藏真实发生的事情,并使调试代码变得不可能。
宏还会绕过预处理器的类型检查,并可能导致运行时问题。
如果您正在尝试优化速度,我会在这里建议使用内联函数。
相关文章:
- 使用std::multimap迭代器创建std::list
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- 错误:调用"std::vector<:vector<int>>::p ush_back(std::vector<std::__cxx11::basic_string<
- C 建造者Clang STD :: Sill,找不到超载的操作员&lt;
- 为什么STD :: MAP需要操作员&lt;以及我如何写一个
- std::vector::reserve(未知m),我知道m<<;N(通常)并且知道N
- std::vector<;uint8_t>;当C++11/14启用时,手动复制而不是调用memcpy
- C++运算符<<调用::ostream而不是std::osttream
- 是std :: set&lt; std :: future&gt;不可能存在
- 在修改列表后,std :: list&lt; t&gt; :: end()的值是否会更改
- BOOST ::变体无法解决运算符&lt;&lt;对于STD :: Ostream
- C++重载<<具有typedef'd std::vector
- 以x的倍数填充前导零,使用std::cout<<std::十六进制
- 错误:没有匹配'运算符<<"在'std::cout
- std::pair的默认构造函数<>将基本类型(int等)设置为零
- std::ostream&operator< & lt; (std:: ostream&压力,压力& &;val)
- 将std::endl传递给std::operator<<
- std::映射<>或std::vector<>在处理大型标志集时
- 重载& lt; & lt;使用命名空间std