有没有办法永久设置std::setw
操纵器(或其功能width
)?看这个:
#include#include #include #include int main( void ) { int array[] = { 1, 2, 4, 8, 16, 32, 64, 128, 256 }; std::cout.fill( '0' ); std::cout.flags( std::ios::hex ); std::cout.width( 3 ); std::copy( &array[0], &array[9], std::ostream_iterator ( std::cout, " " ) ); std::cout << std::endl; for( int i = 0; i < 9; i++ ) { std::cout.width( 3 ); std::cout << array[i] << " "; } std::cout << std::endl; }
跑完后,我看到:
001 2 4 8 10 20 40 80 100 001 002 004 008 010 020 040 080 100
即每个操纵器都保持其位置,除了必须为每个条目设置的setw
/ width
.有没有优雅的方式如何使用std::copy
(或其他东西)setw
?优雅的我当然不是指创建自己的函子或函数来编写东西std::cout
.
嗯,这是不可能的.无法.width
再次拨打电话.但是你当然可以使用boost:
#include#include #include #include #include int main() { using namespace boost::lambda; int a[] = { 1, 2, 3, 4 }; std::copy(a, a + 4, boost::make_function_output_iterator( var(std::cout) << std::setw(3) << _1) ); }
它确实创建了自己的仿函数,但它发生在幕后:)
由于setw
并且width
不会导致持久性设置,因此一种解决方案是定义一个覆盖的类型,在值之前operator<<
应用setw
.这将允许ostream_iterator
该类型的功能std::copy
如下所示.
int fieldWidth = 4; std::copy(v.begin(), v.end(), std::ostream_iterator< FixedWidthVal>(std::cout, ","));
可以定义:(1)FixedWidthVal
与用于数据类型(参数模板类typename
)和宽度(值),和(2)一个operator<<
用于ostream
和一个FixedWidthVal
适用setw
于每个插入.
// FixedWidthVal.hpp #includetemplate struct FixedWidthVal { FixedWidthVal(T v_) : v(v_) {} T v; }; template std::ostream& operator<< (std::ostream& ostr, const FixedWidthVal &fwv) { return ostr << std::setw(W) << fwv.v; }
然后它可以应用std::copy
(或for
循环):
// fixedWidthTest.cpp #include#include #include #include "FixedWidthVal.hpp" int main () { // output array of values int array[] = { 1, 2, 4, 8, 16, 32, 64, 128, 256 }; std::copy(array,array+sizeof(array)/sizeof(int), std::ostream_iterator< FixedWidthVal >(std::cout, ",")); std::cout << std::endl; // output values computed in loop std::ostream_iterator > osi(std::cout, ","); for (int i=1; i<4097; i*=2) osi = i; // * and ++ not necessary std::cout << std::endl; return 0; }
输出(演示)
1, 2, 4, 8, 16, 32, 64, 128, 256, 1, 2, 4, 8, 16, 32, 64, 128, 256, 512,1024,2048,4096,