Class array, an STL Container (as Wrapper) for Arrays of Constant Size |
[intro] [interface] [discussion] [code]
C++ 標準ライブラリの一部である C++ 標準テンプレートライブラリ STL は、さまざまな種類のコンテナに対してアルゴリズムを適用するためのフレームワークを提供している。しかしながら、通常の配列は STL コンテナのインターフェイスには備わっていない(STL コンテナのイテレータとしてのインターフェイスは用意されているけれども)。
通常の配列を置き換えるものとして、STL では vector<>が提供されているが、vector<> は動的配列のセマンティクスを持つので、要素数が変化する可能性を持つデータを管理対象とする。静的なサイズさえあれば充分な場面では、このことはいくぶんかのオーバーヘッドを生じさせることとなる。
Matthew H. Austern は彼の本、Generic Programming and the STL の中で、block と名づけられた、静的サイズの通常の配列を扱うとても便利なラッパクラスを紹介している。これは通常の配列よりも安全であり、パフォーマンスもひけを取らない。The C++ Programming Language, 3rd edition で、Bjarne Stroustrup は c_array という同じようなクラスを紹介しているが、これは私(Nicolai Josuttis)が、私の本 The C++ Standard Library - A Tutorial and Reference の中で carray と呼んでいるものを少々変更して提供したものだ。ここに紹介するのは、これらのアプローチのエッセンスを、boost からのたくさんのフィードバックで味付けしたものである。
いろいろな名前を考えたすえ、このクラスの名前はシンプルに array と決定した。
型: | |
value_type | 要素の型 |
iterator | イテレータの型(ランダムアクセス・イテレータ) |
const_iterator | 定数とされる要素へのイテレータの型 |
reference | 要素の参照の型 |
const_reference | 定数とされる要素への参照の型 |
size_type | 符号付のサイズの型 |
difference_type | 符号なしの距離(差分)の型 |
操作: | |
array<type,num> | デフォルトコンストラクタ。num (要素数)個のtype (型)を要素とする配列を作成する。下のコメントを参照。 |
array<type,num>(a) | コピーコンストラクタa のすべての要素をコピーする。(a は同じ type と num でなければならない)。 |
operator= | 代入。すべての要素を代入する。 |
assign(val) | すべての要素に val を代入する。 |
begin() | 最初の要素へのイテレータを返す。 |
end() | 最後の要素の次の位置へのイテレータを返す。 |
rbegin() | 逆イテレーションでの最初の要素を指す逆イテレータを返す。 |
rend() | 逆イテレーションでの最後の要素のひとつ後ろ(behind)を指す逆イテレータを返す。 |
operator[i] | インデックス i の要素を返す(範囲チェックなし) |
at(i) | インデックス i の要素を返す(i が不正な場合は std::range_error を送出する) |
front() | 最初の要素を返す(要素が存在することを呼び出し側で保証しなければならない) |
back() | 最後の要素を返す(要素が存在することを呼び出し側で保証しなければならない) |
data() | 生の配列を読み取り専用として返す。 |
size() | 要素数を返す。 |
empty() | 配列が空かどうかを返す。 |
max_size() | 格納可能な要素数の最大値を返す(size() と同じ)。 |
swap(a) | array a と要素を交換する。 |
==, != | 等値性を調べる。 |
<, <=, >, >= | array の比較演算。 |
値: | |
static_size | コンパイル時の要素数 |
array クラスは"reversible container"(C++ 標準 Section 23.1, [lib.container.requirements] を参照)の要件のほとんどを満たしているが、完全にではない。array が reversible な STL コンテナではない理由は以下のとおりである。
- 提供されるべきコンストラクタがない
- 要素が不確定な初期値を持つ可能性がある(以下を参照)。
- swap() の計算量が定数ではない。
- size() が常に一定で、型の第2テンプレート引数によって決定される。
- アロケータのサポートを提供しないコンテナである。
また、"sequence"(C++標準Section 23.1.1, [lib.sequence.reqmts]参照)であるための要件を、以下のものを除いて満たしていない。
- front() と back() が提供されている。
- operator[] と at() が提供されている。
コンストラクタの点に関しては、重要な設計上のトレードオフを行い、(それによって)"aggregate"( C++ 標準 Section 8.5.1, [dcl.init.aggr] を参照)として array を実装することができた。つまり、
boost::array<int,4> a = { { 1, 2, 3 } };
リスト内の初期値の数が足りない場合、残りの要素はデフォルト値のままになる(したがって定義された値を持つ)ことに注意しよう。
現在の実装はこのようなアプローチをとっているが、不確定な初期値を持つ可能性があることは大きな問題点である。そこで、この点に関してこのようにすべきだというあなたの考えを、ぜひフィードバックしてほしい。ここから導かれる未解決の問題点は以下のようなものだ。
int data[] = { 1, 2, 3, 4 }
array<int,5> x(data); あるいは array<int,data> x;
規格の 8.5.1 (11) ではこれは許されているようだが、gcc 2.95.2 では警告メッセージを表示する。
建設的なフィードバックはどのようなものでも歓迎する。注意してほしいのは、boostメーリングリストのすべてのメールを読むだけの時間が、私にはないという点だ。というわけで、確実にフィードバックが私に届くようにするため、このクラスに関するメールについては、私にコピーを送ってほしい。
以下のコードは「このままの形(as is)」で提供され、明示的あるいは暗黙的な保証はない。
array.hpp, array<> の実装ファイル:
array1.cpp, array<>を使った簡単な例:
array2.cpp, array<>を使った別の例:
array3.cpp, array<>を使った三つ目の例:
array4.cpp, array<> の array<> を使った例:
array5.cpp, array<> の他の操作をテストする例:
C++ での通常の配列の扱いと、STL フレームワークについてのさらなる詳細について知るには、例えば以下のようなものを参照してほしい。
The C++ Standard Library - A Tutorial and Reference
by Nicolai M. Josuttis
Addison Wesley Longman, 1999
ISBN 0-201-37926-0
[intro] [interface] [discussion] [code]
Japanese Translation Copyright (C) 2003 Tomotoshi Akiyama.
オリジナルの、及びこの著作権表示が全ての複製の中に現れる限り、この文書の
複製、利用、変更、販売そして配布を認める。このドキュメントは「あるがまま」
に提供されており、いかなる明示的、暗黙的保証も行わない。また、
いかなる目的に対しても、その利用が適していることを関知しない。
このドキュメントの対象: Boost Version 1.29.0
オリジナルの最新版はこちら