[解決済み] [Solved] 似たようなconstとnon-constのメンバ関数間のコードの重複を取り除くには?
2022-04-19 08:39:23
質問
例えば、次のようなものがあるとします。
class X
ここで、内部メンバーへのアクセスを返したい。
class Z
{
// details
};
class X
{
std::vector<Z> vecZ;
public:
Z& Z(size_t index)
{
// massive amounts of code for validating index
Z& ret = vecZ[index];
// even more code for determining that the Z instance
// at index is *exactly* the right sort of Z (a process
// which involves calculating leap years in which
// religious holidays fall on Tuesdays for
// the next thousand years or so)
return ret;
}
const Z& Z(size_t index) const
{
// identical to non-const X::Z(), except printed in
// a lighter shade of gray since
// we're running low on toner by this point
}
};
2つのメンバー関数
X::Z()
と
X::Z() const
は、中括弧の中に同じコードがあります。 これは重複したコードです
また、複雑なロジックを持つ長い関数では、メンテナンス上の問題を引き起こす可能性があります。
.
このコードの重複を回避する方法はありますか?
解決方法は?
はい、コードの重複を避けることは可能です。 constメンバ関数でロジックを記述し、非constメンバ関数でconstメンバ関数を呼び出し、戻り値を非const参照(関数がポインタを返す場合はポインタ)に再キャストすればいいのです。
class X
{
std::vector<Z> vecZ;
public:
const Z& z(size_t index) const
{
// same really-really-really long access
// and checking code as in OP
// ...
return vecZ[index];
}
Z& z(size_t index)
{
// One line. One ugly, ugly line - but just one line!
return const_cast<Z&>( static_cast<const X&>(*this).z(index) );
}
#if 0 // A slightly less-ugly version
Z& Z(size_t index)
{
// Two lines -- one cast. This is slightly less ugly but takes an extra line.
const X& constMe = *this;
return const_cast<Z&>( constMe.z(index) );
}
#endif
};
NOTE することが重要です。 NOT のロジックを非 const 関数に置き、const 関数で非 const 関数を呼び出すようにします。 なぜなら、定数クラスのインスタンスは定数でないインスタンスとしてキャストされるからです。 C++の標準では、これは未定義の動作になるとされています。
関連
-
[解決済み】C++エラー。アーキテクチャ x86_64 に対して未定義のシンボル
-
[解決済み】指定範囲内の乱数で配列を埋める(C++)
-
[解決済み] 警告:暗黙の定数変換でのオーバーフロー
-
[解決済み] スタックアロケーションにより初期化されていない値が作成された
-
[解決済み] const int*、const int * const、int const *の違いは何ですか?
-
[解決済み] C#のconstとreadonlyの違いは何ですか?
-
[解決済み] constexpr` と `const` の相違点
-
[解決済み] const "と "val "の違いは何ですか?
-
[解決済み】char * constとconst char *の違いは何ですか?
-
[解決済み】なぜ、constでない参照は一時的なオブジェクトにバインドできないのでしょうか?
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
[解決済み】C++ 非推奨の文字列定数から「char*」への変換について
-
[解決済み】C-stringを使用すると警告が表示される。"ローカル変数に関連するスタックメモリのアドレスが返される"
-
[解決済み】致命的なエラー LNK1169: ゲームプログラミングで1つ以上の多重定義されたシンボルが発見された
-
[解決済み】IntelliSense:オブジェクトに、メンバー関数と互換性のない型修飾子がある
-
[解決済み】C++エラー:の初期化に一致するコンストラクタがありません。
-
[解決済み】オブジェクト引数のない非静的メンバ関数の呼び出し コンパイラーエラー
-
[解決済み】C++の余分な資格エラー
-
[解決済み】指定範囲内の乱数で配列を埋める(C++)
-
[解決済み】 while(cin) と while(cin >> num) の違いは何ですか?)
-
[解決済み] 重複、constとnon-const、ゲッターのエレガントな解決策?重複