
MFCで、CStringをメンバとして含むクラスと、このクラスのオブジェクトを動的に生成して値を代入し、生成したオブジェクトへのポインタを返す関数を以下のように定義しました。
class Record
{
public:
long Id;
CString Str;
};
Record* CreateRecord(long Id_in, CString Str_in)
{
Record* ret;
if ((ret = (Record*)malloc(sizeof(Record))) == NULL) {
return NULL;
}
ret->Id = Id_in; // (1)
ret->Str = Str_in; // (2)
return ret;
}
この関数のコンパイルはうまくいきますが、関数実行時にメモリ参照エラーとなります。
調べてみると、(1)のlong型変数への代入はうまくいっているのですが、(2)のCString型変数への代入がうまくいっていないようです。
既にインスタンス化されているRecord型オブジェクトへのポインタを受け取り、それに代入するという関数であればうまく動きました。
(例)
void SetRecord(Record* received, long Id_in, CString Str_in)
{
received->Id = Id_in;
received->Str = Str_in;
}
先に示したCreateRecord関数は、どこが良くないのでしょうか。
No.2ベストアンサー
- 回答日時:
mallocでメモリを確保しただけでは、
ret->Strのコンストラクタが呼ばれていません。なので、ret->Strはめちゃくちゃな中身になっています。というわけで、ret->Strにアクセスすると落ちます。
C++で動的にメモリを使いたいときは、よっぽど特殊な事情がない限り基本的にはmallocではなくてnewを使ってください。
ご回答いただき、ありがとうございます。
なるほど、確かにmallocを使うと必要な量のメモリは確保されますが、実際にインスタンス化されたわけではなく、データ同士の相互関係等が未定義のままですね。
まだオブジェクト指向言語の経験が浅いので、もっと基礎を固めたいと思います。
No.3
- 回答日時:
C++ではnewを使うというのは他の回答者さんの書かれている通り。
(あわせて、freeの代わりにdeleteを使うことも忘れずに)
その上で、現行のC++ではnewの確保失敗は単純にNULLが返るわけではありませんので、
C言語のようなNULLとの比較では意図通りに動きません。
例外処理を調べてtryを使ってください。
※コンパイラのバージョンや設定にもよります。
例えば現行のC++仕様より古いVC6ならNULL比較で正しく動きます。
また、MFCには独自クラスも用意されていますので、
std::bad_allocやCMemoryExceptionをcatchします。
ご回答ありがとうございます。
C++ではメモリアロケーションエラーは例外処理を使えばいいのですね。
とても参考になります。
オブジェクトの削除もdeleteを使うようにします。
No.1
- 回答日時:
mallocでは領域を確保したに過ぎないのではないかと思います
Record* CreateRecord(long Id_in, CString Str_in)
{
Record* ret;
if ((ret = new Record()) == NULL) {
return NULL;
}
ret->Id = Id_in; // (1)
ret->Str = Str_in; // (2)
return ret;
}
といった具合に new演算子をつかってRecordクラスのコンストラクタを呼んでみましょう
ご回答いただき、ありがとうございます。
教えていただいたとおりnew演算子を使ってみたところ、うまく実行できました。
Cでのmallocは、C++ではnewを使えばいいのですね。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
このQ&Aを見た人はこんなQ&Aも見ています
関連するカテゴリからQ&Aを探す
おすすめ情報
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
allocってなんですか?
-
LoadLibraryでAccess Violation...
-
座標値を読み込んである領域に...
-
gcnew arrayによるメモリ確保に...
-
構造体でchar name[]と*nameの...
-
void*型のデータサイズ
-
DLLで同じメモリ領域を参照する...
-
win32APIのHeapAlloc()の使い方...
-
LPWSTRのコピー
-
C++で、メンバもヒープに確保さ...
-
Win32APIでのメモリ管理について
-
DLLのマルチスレッドの動作につ...
-
mallocで確保するメモリの領域...
-
CreateFileMapping について
-
c言語のポインタへの文字列入力...
-
CStringからchar*への型変換に...
-
C言語 配列の長さの上限
-
char*を初期化したいのですが
-
init関数の意味
-
System.IO.Directory.GetFiles...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
allocってなんですか?
-
c言語のポインタへの文字列入力...
-
HEAP に関すること
-
構造体でchar name[]と*nameの...
-
ビットをローテートするプログ...
-
ヒープメモリの解放について
-
DLLのマルチスレッドの動作につ...
-
グローバル変数のサイズ
-
C++で、メンバもヒープに確保さ...
-
newしないオブジェクトについて
-
free関数で動作が止まる
-
構造体配列の初期化について
-
void*型のデータサイズ
-
MFCのCStringについて
-
mallocで確保するメモリの領域...
-
CreateFileMapping について
-
LPWSTRのコピー
-
c言語のメモリの確保について
-
配列の添え字の最大数とは?
-
C言語の質問です。 以下の命令...
おすすめ情報