ホームページ >Java >&#&チュートリアル >Java のコピー コンストラクターの例の詳細な説明
タイトル: 次の整数配列クラスの定義には、各関数に 1 つだけエラーがあります。見つけて修正してください。
class ArrayInt { public: ArrayInt(const int *pArray, int size) : m_nSize(size) { assert(size > 0); for (int i = 0; i < size; ++ i) m_pArray[i] = pArray[i]; } ArrayInt(const ArrayInt &iCopy):m_nSize(iCopy.m_nSize),m_pArray(iCopy.m_pArray){} ~ArrayInt() { delete m_pArray; } int operator[](short index) { assert(index < m_nSize); return m_pArray[index]; } const ArrayInt & operator=(const ArrayInt & iCopy) { assert(m_nSize >= iCopy.m_nSize); for (int i = 0; i < m_nSize; ++ i) m_pArray[i] = iCopy.m_pArray[i]; } private: short m_nSize; int * m_pArray; };
分析: 上記エラー これらは C の最も基本的な知識であり、最も混乱を招く部分です コピー コンストラクターに関して注意すべき点の 1 つは、ディープ コピーとシャロー コピーの問題です。上記の関数のそれぞれの問題は次のように説明されます:
1. コンストラクター内にスペースを割り当てずに代入が開始されます;
2. コピー コンストラクターは浅いコピーであるため、2 つのオブジェクトがピースを共有します。メモリ
3. 配列メモリはデストラクタ内で削除され、ポインタには空の値が割り当てられる必要があります
#4. 括弧演算子関数の添字範囲外チェックが不完全ですインデックスが負の値の場合、プログラムはクラッシュします。#5. 代入演算子関数には戻り値がありません。戻り値の機能は、a = b = c;
## を継続的に代入することです。 #プログラムの修正は次のとおりです:class ArrayInt { public: ArrayInt(const int *pArray, int size) : m_nSize(size) { assert(size > 0); m_pArray = new int[size]; for (int i = 0; i < size; ++ i) m_pArray[i] = pArray[i]; } ArrayInt(const ArrayInt & iCopy) { //ArrayInt(iCopy.m_pArray, iCopy.m_nSize); m_nSize = iCopy.m_nSize; assert(m_nSize > 0); m_pArray = new int[m_nSize]; for (int i = 0; i < m_nSize; ++ i) m_pArray[i] = iCopy.m_pArray[i]; } ~ArrayInt() { if (m_pArray) { delete[] m_pArray; m_pArray = NULL; } //printf("distructor is called\n"); } int operator[](short index) { assert(index < m_nSize && index >= 0); return m_pArray[index]; } const ArrayInt & operator=(const ArrayInt & iCopy) { if (this == &iCopy) return *this; assert(m_nSize >= iCopy.m_nSize); for (int i = 0; i < iCopy.m_nSize; ++ i) m_pArray[i] = iCopy.m_pArray[i]; return *this; } private: short m_nSize; int * m_pArray; };
注:
コピー コンストラクターでは、コンストラクターを呼び出してディープ コピーを実装することはできません。理由は、匿名オブジェクトであるためです。コピー コンストラクターが呼び出された後、オブジェクトが破棄されるため (デストラクター内の文字を出力することで確認できます)、ディープ コピーは期待どおりに実装されません。匿名オブジェクトはディープ コピーされます。したがって、コピー コンストラクターを呼び出すオブジェクトのデータ メンバーに外部からアクセスすると、エラーが報告されます。テスト関数は次のとおりです:
void test_construct_copy() { int pArray[] = {1, 2, 3, 5}; ArrayInt arr(pArray, sizeof pArray / sizeof(int)); printf("%d \n", arr[2]); ArrayInt arr2(arr); printf("%d \n", arr2[2]); pArray[2] = 8; ArrayInt arr3(pArray, 4); printf("%d \n", arr3[2]); arr3 = arr2; printf("%d \n", arr3[2]); pArray[2] = 10; ArrayInt arr4(pArray, 4); arr3 = arr2 = arr4; printf("%d \n", arr3[2]); }
以上がJava のコピー コンストラクターの例の詳細な説明の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。