当前位置:  开发笔记 > 编程语言 > 正文

"永久"std :: setw

如何解决《"永久"std::setw》经验,为你挑选了2个好方法。

有没有办法永久设置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.



1> Johannes Sch..:

嗯,这是不可能的.无法.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)
        );
}

确实创建了自己的仿函数,但它发生在幕后:)



2> chappjc..:

由于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
#include 

template 
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,


@j_random_hacker好吧,我应该在信用到期时给予信任.我从[codereview question](http://codereview.stackexchange.com/q/18291/35254)采用了这种方法,只是添加了数据类型模板参数.方便功能的好建议.
非常好的设计,我认为适用于很多情况.如果宽度可以是运行时(而不是编译时)参数,那将是理想的,尽管我无法想到将这些信息"转换为"ostream_iterator`的好方法.你还可以提供一个方便函数`template with_width(T v){return FixedWidthVal (v,width); 保存必须指定类型.
推荐阅读
李桂平2402851397
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有