ホームページ > バックエンド開発 > C++ > C で継承階層のベクトルを扱うときにオブジェクトのスライスを回避するにはどうすればよいですか?

C で継承階層のベクトルを扱うときにオブジェクトのスライスを回避するにはどうすればよいですか?

DDD
リリース: 2024-11-01 09:00:03
オリジナル
950 人が閲覧しました

How Can Object Slicing Be Avoided When Working with Vectors of Inheritance Hierarchies in C  ?

C のベクトルとポリモーフィズム : 解明されたオブジェクト スライシング

C で継承階層のベクトルを扱うとき、次のような微妙な複雑さに遭遇することがあります。意図した動作を妨げます。次の例を考えてみましょう。

<code class="cpp">class Instruction
{
public:
    virtual void execute() {  }
};

class Add: public Instruction
{
private:
    int a;
    int b;
    int c;
public:
    Add(int x, int y, int z) {a=x;b=y;c=z;}
    void execute() { a = b + c;  }
};</code>
ログイン後にコピー

次に、基本クラス ポインターのベクトルに Add オブジェクトを格納することを想像してください。

<code class="cpp">void some_method()
{
    vector<Instruction*> v;
    Instruction* i = new Add(1,2,3)
    v.push_back(i);
}</code>
ログイン後にコピー

その後、別のメソッドで、ベクトルの最後の要素で関数を実行します:

<code class="cpp">void some_other_method()
{
    Instruction ins = v.back();
    ins.execute();
}</code>
ログイン後にコピー

よくある誤解は、実行関数は Add オブジェクトで呼び出されたかのように動作するというものです。しかし、これは当てはまりません。

この問題は、ベクトルが値を格納する方法に起因します。つまり、ベクトルは参照ではなくオブジェクトのコピーを格納します。したがって、Add オブジェクトをベクターにプッシュするとコピーされ、その結果、命令タイプのオブジェクトが生成され、Add 固有のデータとメソッドが失われます。これはオブジェクトのスライスとして知られています。

オブジェクトのスライスを回避するには、コピーの代わりに参照を保持するようにベクトルを変更する必要があります。

<code class="cpp">vector<Instruction*> ins</code>
ログイン後にコピー

あるいは、std::reference_wrapper を使用することもできます。

<code class="cpp">vector< std::reference_wrapper<Instruction> > ins</code>
ログイン後にコピー

参照を使用することで、ベクター内でオブジェクトの元の型が保持され、期待どおりの多態性の動作が可能になります。

以上がC で継承階層のベクトルを扱うときにオブジェクトのスライスを回避するにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート