web-dev-qa-db-ja.com

C ++のストリームマニピュレータを使用してテキストを固定幅フィールドの中央に配置します

printfをlong文字列(実際のフォーマットなし)で使用して、概念的に次のようなプレーンテキストのテーブルヘッダーを出力するレガシーコードをリファクタリングしています。

|  Table   |  Column  | Header  |

現在、次のように作成されています。

printf("|  Table   |  Column  | Header  |");

上記を次のようなコードで作成したいと思います。1

outputStream << "|" << std::setw(10) << std::center << "Table"
             << "|" << std::setw(10) << std::center << "Column"
             << "|" << std::setw(9) << std::center << "Header"
             << "|" << std::endl;

<iomanip>にはストリームマニピュレータstd::leftstd::right、およびstd::internalがあるため、コンパイルされませんが、std::centerはないようです。これを標準C++ライブラリですでに実行するクリーンな方法はありますか、それとも必要な間隔を手動で計算する必要がありますか?


1これはCコードよりも冗長ですが、printfステートメントの数と文字列内の固定された重複の量のため、長期的には冗長性が低くなります。また、より拡張性が高く、保守が容易になります。

14
Keith Pinson

これがあなたが望むことを達成するヘルパークラスです:

_#include <string>
#include <iostream>
#include <iomanip>

template<typename charT, typename traits = std::char_traits<charT> >
class center_helper {
    std::basic_string<charT, traits> str_;
public:
    center_helper(std::basic_string<charT, traits> str) : str_(str) {}
    template<typename a, typename b>
    friend std::basic_ostream<a, b>& operator<<(std::basic_ostream<a, b>& s, const center_helper<a, b>& c);
};

template<typename charT, typename traits = std::char_traits<charT> >
center_helper<charT, traits> centered(std::basic_string<charT, traits> str) {
    return center_helper<charT, traits>(str);
}

// redeclare for std::string directly so we can support anything that implicitly converts to std::string
center_helper<std::string::value_type, std::string::traits_type> centered(const std::string& str) {
    return center_helper<std::string::value_type, std::string::traits_type>(str);
}

template<typename charT, typename traits>
std::basic_ostream<charT, traits>& operator<<(std::basic_ostream<charT, traits>& s, const center_helper<charT, traits>& c) {
    std::streamsize w = s.width();
    if (w > c.str_.length()) {
        std::streamsize left = (w + c.str_.length()) / 2;
        s.width(left);
        s << c.str_;
        s.width(w - left);
        s << "";
    } else {
        s << c.str_;
    }
    return s;
}
_

次のように、centered("String")を呼び出すだけで使用されます。

_int main(int argc, char *argv[]) {
    std::cout << "|" << std::setw(10) << centered("Table")
              << "|" << std::setw(10) << centered("Column")
              << "|" << std::setw(9)  << centered("Header") << "|"
              << std::endl;
}
_
10
Lily Ballard

std::centerマニピュレータはありません。私はあなたがそれを自分でしなければならないのではないかと心配しています。作業を減らすために、幅と文字列を指定してスペースを計算するヘルパー関数を作成できます。

ヘルパー関数がどのように見えるかのサンプルを次に示します。それをより効率的にするためにいくつかの作業が必要です。

string helper(int width, const string& str) {
    int len = str.length();
    if(width < len) { return str; }

    int diff = width - len;
    int pad1 = diff/2;
    int pad2 = diff - pad1;
    return string(pad1, ' ') + str + string(pad2, ' ');
}
4
Masked Man

手動で行う必要があります。しかし、文字列を扱う場合はそれほど難しくありません。何かのようなもの:

std::string
centered( std::string const& original, int targetSize )
{
    assert( targetSize >= 0 );
    int padding = targetSize - checked_cast<int>( original.size() );
    return padding > 0
        ? std::string( padding / 2, ' ' ) 
            + original
            + std::string( targetSize - (padding / 2), ' ' )
        : original;
}

トリックを行う必要があります。

3
James Kanze

nCURSESライブラリを使用できます...それはちょっとファンクで、コンパイルコマンドに「-L NCURSES」を追加する必要があります(g ++ -L NCURSES yourProgram.cpp)。それは機能し、色やその他の「クール」を実行できます。もの[とにかくCLIのためにかっこいい]。マニュアルを読むだけで、センタリングは非常に簡単です。 http://tldp.org/HOWTO/NCURSES-Programming-HOWTO/

0
jEXEy