
C++でプログラムを書いていたところメモリリークしているのを発見しました。
そこでなんとかメモリリークしている箇所を特定してみたのですが、なんとstd::wstringとstd::stringがメモリリークしているようです。
(UNICODEとマルチバイト文字に対応させるため両方でプログラムかいています)
ローカル変数で関数さえ抜ければメモリからは消滅するはずのstd::wstringやstd::stringがメモリリークしているのは何か原因でもあるのでしょうか?
よろしくおねがいします。
No.2ベストアンサー
- 回答日時:
こんばんは。
>>ローカル変数で関数さえ抜ければメモリからは消滅するはずのstd::wstringやstd::stringがメモリリークしているのは何か原因でもあるのでしょうか?
原因は幾つか考えられます。そもそもstd::wstring等は、ローカル変数で有ろうと無かろうと、コンテナ内部で動的にメモリのアロケーションを行っているのです。
なので、デストラクタが立ち上がる迄はコンテナ内部のメモリは開放されません。
main()の一番下に_CrtDumpMemoryLeaks()を置いても検出の方が先に実行されます。よってリークしていると判断されてしまいます。
void main(void)
{
Data dat;
dat.age = 35;
dat.name = L"てすと";
hoge( dat );
int leaks = _CrtDumpMemoryLeaks();
//--------------------------------
dat::~Data();の呼び出される。ここでstd::stringの内部が開放される
}
例えDataのデストラクタに_CrtDumpMemoryLeaks()を置いても、その後にstd::stringのデストラクタが立ち上がるので、検出が早すぎる事になります。
以下の物で試して見れば分かる筈です。
class inner
{
public:inner() : p(new int(1)){}
~inner()
{
delete p;
cout << "innter destructor : " << _CrtDumpMemoryLeaks() << endl;//ココが適切な位置
}
private:int* p;
};
class outer
{
public:~outer()
{cout << "outer destructor : " << _CrtDumpMemoryLeaks() << endl;}//早すぎる
private:inner in;
};
int main(void)
{
cout << "in main function " << _CrtDumpMemoryLeaks() << endl;//早すぎる
return 0;
}
まあ、この程度なら良いのですが、もっと沢山のクラスが有る場合、追跡するのは非常に大変です。ただ、せめても、デストラクタだけは書いておいた方が、やり易くはなるとは思います・・・。
で、質問者様のコードは、以下の様にすればリークしていないのが分かります。
void test()
{
Data dat;
dat.age = 35;
dat.name = L"てすと";
hoge( dat );
}
int main(void)
{
test();
cout << "in main function " << _CrtDumpMemoryLeaks() << endl;
return 0;
}
此れでリークしていた場合
(1)別の何処かでリークしている
(2)STLのアロケータの問題
と言う事に成りそうです。(1)は良いとして、(2)の方はSTLPort等を使用している場合、デフォルトアロケータ(SGIアロケータ)を使用していると、容赦なくリークします。
http://www.google.co.jp/search?hl=ja&q=STLPort+S …
その場合、コンフィグのマクロで、new又はmallocで割り当てるアロケータを使用させます。
回答ありがとうございます。
デストラクタよりも先にメモリリークの検出が行われてしまうのですね。
関数を1つはさんでみたところ見事にメモリリークは消えました。
ありがとうございました。
No.1
- 回答日時:
> なんとstd::wstringとstd::stringがメモリリークしているようです。
間違いなくそうであるかどうかを念のために確認したいので、
お書きになったコード全体を見せてくださいますか?
ていうか、コードの中身と起きている現象とを最初からセットで見せてくだされば、
こんな回答しなくてすむんですけれどね。
この回答への補足
回答ありがとうございます。
コード全体といいましてもちょっとここに載せるには規模が大きすぎるのでちょっと載せることはできませんのでこんな感じのことをやっているということだけ。
struct Data
{
int age;
std::string name
};
class Test
{
public:
Data dat;
};
void hoge( Data dat )
{
Test *p = new Test
p->dat = dat;
delete p;
}
void main(void)
{
Data dat;
dat.age = 35;
dat.name = L"てすと"
hoge( dat );
}
のように構造体のメンバ変数にstringを格納して構造体ごと関数に渡しているだけです。
hogeの中でクラスインスタンスを生成して、そのインスタンスのメンバ変数に直接データを流して込んでいます。
わかりづらくて申し訳ありません。
ただローカル変数がメモリリークというのは初めてでして。よろしくお願いします。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
このQ&Aを見た人はこんなQ&Aも見ています
関連するカテゴリからQ&Aを探す
おすすめ情報
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
なぜ、C++の標準ヘッダをインク...
-
DLLでダイアログ
-
ヘッダーファイルがインクルー...
-
CStringとString
-
組み合わせと順列 アルゴリズム
-
winpcapを用いたプログラミング
-
構文エラーが出ているのですが...
-
VC++で文字列から任意の文字を...
-
C言語のポインターで詰まっている
-
vectorのイテレータを大小比較...
-
string型の固定長文字列を配列...
-
C++で、テキストファイルを一行...
-
C++でのmath.hやstdio.hの使用...
-
opencvを使って画像比較
-
VHDLのvector
-
VxWorks 6.4ソケット接続につい...
-
コンパイルエラー
-
C++で2次元配列charをループしたい
-
リモートデスクトップの接続元I...
-
条件付き取り込み
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
VC++で文字列から任意の文字を...
-
なぜ、C++の標準ヘッダをインク...
-
構文エラーが出ているのですが...
-
enumの値から定義名を文字列化...
-
JPEGやPNGが読めるLoadImage関数
-
std::map の const 修飾について
-
VS2019でofstreamが未定義になる
-
_tcscat がうまくいきません(V...
-
空ENTERの判別
-
構造体配列のvectorへの変換と...
-
switch文のエラーについて
-
std::wstringのメモリリークに...
-
#define中の#のエスケープ
-
C++でShowCursorを使いたい。
-
findnext();について
-
【C++】ヘッダ内でstringを格納...
-
#include "fstream.h"
-
CStringとString
-
#defineの使い方について
-
iostream インクルード時に発生...
おすすめ情報