[C++] Вопрос по распределению памяти
И что ты передаешь в конструктор?
Конструктор объявлен так:
MyClass(long, constr char*)Создаю так
MyClass obj(28, "string containing 28 chars");
Учитесь пользоваться отладчиком, это еще никому не повредило.
Да, поставь брейкпойнт в деструкторе. Не считая грамматической ошибки в конструкторе ничего подозрительного.
А нет, вру. Если в функции создаю ссылку на объект, то функция отрабатывает нормально каждый раз, но результат (с одними и теми же данными, то есть аргументами переданными в конструктор) рандомный, хотя класс создаётся исходя только из аргументов, переданных в конструктор.
Вообще, конструктор объявлен так:
MyClass(long size, const char* str)
* это указатель. & это ссылка. Так ссылку или указатель?
Где-то в объекте класса хранится левый указатель, и освобождается в деструкторе. Проверять надо, скорее всего, как раз конструктор. А вообще, телепаты вроде как в отпуске опять, было бы неплохо увидеть код класса.
Класс зовётся wbenData. Там используется парочка классов из wxWidgets.
Не считая грамматической ошибки в конструкторе ничего подозрительного.Лично мне кажется подозрительным, что в нагрузку к const char * передается длина.
деструктор такого не любит, но в версии с * до него даже не доходит - память течь будет
осмотри класс на предмет вышеуказанных опасностей, подебажь деструктор в той версии, которая падает.
На какой строчке падает?
bool wtfTorrentTab::ProcessSingleВылетает, по-видимому, при выходе из функции — сообщение выдаётся, а после того, как нажмёшь кнопку "Ok", выдаётся сообщение винды с предложением отослать отчёт..
{
wbenData data(3, "i1e");
int error = data.error;
wxString msg;
switch (error)
{
case 0:
break;
case W_BENCODE_ERROR_UNKNOWN_DATA_TYPE:
msg = wxT("Error: Unknown data type");
break;
case W_BENCODE_ERROR_UNEXPECTED_TOKEN:
msg = wxT("Error: Unexpected token");
break;
case W_BENCODE_ERROR_NO_END_TAG:
msg = wxT("Error: No end tag");
break;
case W_BENCODE_ERROR_ODD_ELEMENTS_IN_DICTIONARY:
msg = wxT("Error: Odd element in dictionary");
break;
case W_BENCODE_ERROR_INVALID_KEY:
msg = wxT("Error: Invalid key (should be a string)");
break;
default:
msg.Printf(wxT("Error: Unknown error %d" error);
break;
}
if (error == 0)
{
msg = DumpData(data);
}
wxMessageBox(msg, wxT("Bencode" wxOK);
return true;
}
wxString wtfTorrentTab::DumpData(wbenData data)
{
wxString res;
wbenDataDictionary::iterator it;
wbenDataDictionary *dict;
wbenDataList *arr;
size_t i, j;
switch (data.GetType
{
case wbenDATA_UNDEFINED:
res = wxT("undef");
break;
case wbenDATA_INTEGER:
res.Printf(wxT("%d" *(data.GetValue.m_int;
break;
case wbenDATA_STRING:
res = wxT("'") + *(data.GetValue.m_str) + wxT("'");
break;
case wbenDATA_LIST:
res = wxT("[");
arr = data.GetValue.m_list;
for (i = 0; i < arr->Count; i++)
{
res += wxT("\n");
res += DumpData(arr->Item(i;
res += wxT(",");
}
res += wxT("]");
break;
case wbenDATA_DICTIONARY:
res = wxT("{");
dict = data.GetValue.m_dict;
for (it = dict->begin; it != dict->end; ++it)
{
res += wxT("\n");
res += wxT("'") + it->first + wxT("' => ");
res += DumpData(*(it->second;
res += wxT(",");
}
res += wxT("}");
break;
default:
res = wxT("Unknown data type");
break;
}
return res;
}
Вылетает, по-видимому, при выходе из функции — сообщение выдаётся, а после того, как нажмёшь кнопку "Ok", выдаётся сообщение винды с предложением отослать отчёт..у меня такое недавно было когда переносил подсистему с ANSI на Unicode. В одном месте забыл передать меньшую длину (в два раза функция wcscpy затирала много лишней памяти и в том числе запарывала стек. Очень неприятная ошибка, трудно найти причину. Хотя здесь и в деструкторе может быть ошибка. Попробуй заключить весь код этой функции в фигурные скобки, за ними же оставь MessageBox и return.
Если не можешь поставить брейкпойнт, то можно после каждой строчки что-нибудь выводить и так понять, где падает. Хотя это не очень хороший способ.
wbenData data(3, "i1e");ты помнишь, что тут строка длины 4, а не 3?
ты помнишь, что тут строка длины 4, а не 3?массив длины 4, строка - 3
Я просто пишу в Far' е, а для компиляции пользуюсь makefile'ом.
массив длины 4, строка - 3такое деление - это все от лукавого...
главное, что какой-нибудь strcpy - поменяет 4 байта, а не 3
пишу чисто для себя (для своего удовольствия)вот это в некотором роде не согласуется с последним постом.
главное, что какой-нибудь strcpy - поменяет 4 байта, а не 3ну а strlen вернет 3, а не 4 =)
Параметр size у меня используется только внутри класса и никуда более не передаётся. А смысл его такой, что обрабатываются первые size байт строки str.
DumpData(wbenData data)Попробуй поставить там wbenData& data
wbenData& dataсначала const wbenData& data, потом если не откомпилиться, подумать.
сначала const wbenData& data, потом если не откомпилиться, подумать.Нет, сразу wbenData& data, потому что надо смотреть котт автора.
Нет, сразу wbenData& dataНу давай, советуй-советуй, я мешать не буду.
Объяснишь, в чём ошибка?
Пробовал закомментить строки, где DumpData вызывала себя рекурсивно, но ошибка возникала всё равно...
смотреть котт автораИ что ты там такого увидел. Что Get-методы не описаны как const?
Объяснишь, в чём ошибка?Если заработало, значит, была проблема в копировании данных. Ты передавал класс wbenData по значению, это приводит к полному копированию его данных. При этом при выходе из функции вызывается деструктор, который удаляет все указатели. Поскольку указатели одинаковые в оригинале и в копии, деструктор оригинала пытается получить доступ к уже освобождённой памяти.
Тебе стоит прочитать что-нибудь по теме конструкторов копирования.
Оставить комментарий
Serega009
Может, некорректно сформулировал тему...Вопрос в следующем.
У меня есть функция: Правильно ли я понимаю, что после её отработки будет вызван деструктор MyClass' а?
А что происходит если функцию модифицировать следующим образом: ?
На самом деле, я написал свой класс, но в первом варианте программа "выполняет недопустимую операцию", а во втором варианте функция один раз отрабатывает нормально (без ошибок а во второй раз вообще не отрабатывает (программа вылетает).
Не знаю, что ещё тут сказать... могу выложить исходник всего проекта (около 150 кило) и описать вкратце как он устроен.
ЗЫ
Не издевайтесь, я в C++ не сильно шарю — пишу чисто для себя (для своего удовольствия).