boost::ptr_map の使い方

boost::ptr_mapの使い方が,どこを探してもはっきりとは載っていなかったのでメモ。 boostのバージョンは1.41。 必要最低限の使い方です。

下準備

boost::ptr_mapを使うためには,boost/ptr_container/ptr_map.hpp をインクルードする必要がある。

#include <boost/ptr_container/ptr_map.hpp>

また,説明するためにこんなクラスを用意。

/**
 * 基底クラス。
 * 純粋仮想関数を含んでいるのでインスタンスかはできない(インターフェイス)。
 */
class BaseContainer {
public:
    virtual void mes() = 0;
};

/** * コンテナクラス。 * BaseContainerを継承。 * 値の保持と,何を保持しているかを表示するメソッド。 */ template <class T> class Container : public BaseContainer { private: T value;

public: Container(T value) { this->value = value; }

<span class="synType">void</span> mes() {
    std::cout &lt;&lt; <span class="synConstant">&quot;Container : value = &quot;</span> &lt;&lt; value &lt;&lt; std::endl;
}

};

boost::ptr_mapインスタンスの生成

キーをstd::string,値を BaseContainer としてインスタンスを生成。 値を BaseContainer にすることによって, Container インスタンスにどんな型を指定しても おなじ ptr_map インスタンスに突っ込めるっていうすんぽー。

boost::ptr_map<std::string, BaseContainer> Map m;

値の追加

生成した ptr_map インスタンスに値を追加する。 ptr_map はキー・値ともに参照で渡さないといけないらしいので, 一度変数に落としてから渡すようにする。 インスタンスを渡す場合には直接 new してもOK。 直接 new しないと実行時にエラーになります。 元のインスタンスと,ptr_mapに追加したインスタンス(両方とも同じ場所を指してる)が, 終了時に両方とも解放されるのが原因なんじゃないかと。

// 一度変数に落とさないとコンパイルエラー
int key1 = 0;
int key2 = 1;

// ポインタとして宣言して new する // これをやると実行時に // error for object 0x7fff5fbff260: pointer being freed was not allocated // のようなエラーが出る(コンパイルは通る)。 // Container<int>* c; // c = new Container<int>(10); // m.insert(id1, c);

// 直接 new する m.insert(id2, new Container<float>(20));

値の取り出し

boost::ptr_map インスタンスからオブジェクトを取り出す場合, std::map と同じように [] を使ってもうまくいかない。 扱ってるのがポインタだから?

// これだとコンパイルエラー
m[key1]

そこでメンバ関数の find() を使ってイテレータを取得してアクセスする。

// キー
m.find(key1)->first;

// 値 m.find(key1)->second;

// Container.mes() の呼び出し m.find(key1)->second->mes();

値の削除

キーを指定して値を削除。 メモリの解放とかは勝手にやってくれるらしい。

// key1 の値を削除
m.erase(key1);

コードの全貌

今回のコードをつなげるとこんな感じになる。

#include <iostream>
#include <string>

#include <boost/ptr_container/ptr_map.hpp>

class BaseContainer { public: virtual void mes() = 0; };

template <class T> class Container : public BaseContainer { private: T value;

public: Container(T value) { this->value = value; }

<span class="synType">void</span> mes() {
    std::cout &lt;&lt; <span class="synConstant">&quot;Container : value = &quot;</span> &lt;&lt; value &lt;&lt; std::endl;
}

};

int main() { typedef boost::ptr_map<int, BaseContainer> m;

std::string key1 = <span class="synConstant">&quot;n1&quot;</span>;
std::string key2 = <span class="synConstant">&quot;n2&quot;</span>;

m.insert(key2, <span class="synStatement">new</span> Container&lt;<span class="synType">int</span>&gt;(<span class="synConstant">98</span>));
m.insert(key2, <span class="synStatement">new</span> Container&lt;<span class="synType">float</span>&gt;(<span class="synConstant">123.4</span>));

std::cout &lt;&lt; m.size() &lt;&lt; std::endl;

std::cout &lt;&lt; m.find(key1)-&gt;first &lt;&lt; std::endl;
m.find(key1)-&gt;second-&gt;mes();

m.erase(key1);
std::cout &lt;&lt; m.size() &lt;&lt; std::endl;

<span class="synStatement">return</span> <span class="synConstant">0</span>;

}

出力

2
0
Container : value = 98
1