Освобождение памяти в деструкторе шаблонного класса, muxas, 02-Авг-05, 16:47 [смотреть все]Здравствуйте,Есть класс, реализующий динамический массив. Предполагается хранить в нем указатели на любые динамические структуры, которые создаются вызовом new() ВНЕ этого класса. Чтобы не зависеть от типа указателя, есть смысл написать этот класс в виде шаблона. Вопрос в том, как особождать память, которую занимают созданные объекты. Очень удобно было бы автоматически зачищать ее ВНУТРИ деструктора шаблонного класса. Такое возможно? Если да, то как? Ниже мой вариант. Для проверки зачистки я использую модуль MemWatch, так он ругается на всех выделениях памяти, и говорит, что память unfreeed. Поэтому и возник этот вопрос. Очень нуждаюсь в помощи - работа стоит, т.к. может всю базу нахрен нужно переписывать. template <class AnyType> class DynArray { private: AnyType *items; // сам массив long int ntotal; // всего выделено памяти long int ncurrent; // всего занято памяти long int nstep; // шаг выделения public: DynArray() {}; DynArray(long int ArraySizeExpansion); ~DynArray(); AnyType operator[](long int index); int append(AnyType item); void remove(long int index); long int length() { return ncurrent; }; enum { NO_ERRORS=0, NOT_ENOUGH_MEMORY }; }; template<class AnyType> DynArray<AnyType>::DynArray(long int ArraySizeExpansion) { ntotal = nstep = (ArraySizeExpansion==0)?1:ArraySizeExpansion; ncurrent = 0; items = (AnyType*) malloc(sizeof(AnyType)*nstep); << MemWatch материться на этой строке memset(items,0,sizeof(AnyType)*nstep); }; template<class AnyType> DynArray<AnyType>::~DynArray() { for (long int i=0;i<ncurrent;i++) delete items[i]; //<< очень удобный вариант зачистки free(items); }; template<class AnyType> AnyType DynArray<AnyType>::operator[](long int index) { return items[index]; }; template<class AnyType> int DynArray<AnyType>::append(AnyType item) { if (ncurrent>=ntotal) { AnyType *newitems = (AnyType*)realloc(items,sizeof(AnyType)*(ntotal+nstep)); if (newitems==NULL) { return NOT_ENOUGH_MEMORY; }; items = newitems; memset(&items[ncurrent],0,sizeof(AnyType)*nstep); ntotal+=nstep; items[ncurrent] = item; ncurrent++; return NO_ERRORS; }; template<class AnyType> void DynArray<AnyType>::remove(long int index) { delete items[index]; //<< здесь тоже memset(items+ncurrent-1,0,sizeof(AnyType)*(ntotal-ncurrent+1)); ncurrent--; };
|
- Освобождение памяти в деструкторе шаблонного класса, Maxim Kuznetsov, 18:08 , 02-Авг-05 (1)
в чём разница приведённого фрагмента и std::vector<AnyClass> myarr ? лучше досочините SmartPointer ;-)
- Освобождение памяти в деструкторе шаблонного класса, muxas, 12:47 , 03-Авг-05 (2)
Если честно, то не знаю, в чем разница, т.к. не смотрел исходников vector'a. Свой написал только потому, что не хочу пользоваться stdlib - растет размер и падает скорость. Так по утечке памяти что-то можно сказать исходя из приведенного кода?
- Освобождение памяти в деструкторе шаблонного класса, Maxim Kuznetsov, 13:27 , 03-Авг-05 (3)
вообще по коду - бред срочно читайте книги на предмет понятий динамической памяти,указателей и массивов
- Освобождение памяти в деструкторе шаблонного класса, muxas, 13:58 , 03-Авг-05 (4)
>вообще по коду - бред Поконкретнее плиз :-)
- Освобождение памяти в деструкторе шаблонного класса, Maxim Kuznetsov, 14:32 , 03-Авг-05 (5)
>>вообще по коду - бред > >Поконкретнее плиз :-) 1) если предполагаеся хранить указатели, то тип массива: AnyType ** (в принципе далее можно и не комментировать) 2) последовательность действий (в append мелькала) memset(&obj,0,sizeof(obj)); obj=another_obj; приводит к краху системы 3) append - это вообще шедевр! функция добавляет элемент, но куда(позицию/индекс/ключ), она никому не скажет, потому что неположенно ;-)
- Освобождение памяти в деструкторе шаблонного класса, muxas, 15:02 , 03-Авг-05 (6)
>1) если предполагаеся хранить указатели, то тип массива: AnyType ** Все немного проще :-) typedef DynArray<SQLParam*> SQLParamArray; >>2) последовательность действий (в append мелькала) >> memset(&obj,0,sizeof(obj)); >> obj=another_obj; >> приводит к краху системы Вы невнимательны. Нули проставляются только в добавленной памяти (realloc'ом), а не в уже занятой. Указатель на объект присваивается первой пустой ячейке в массиве, т.к. ncurrent указывает не на последний занятый элемент, а на первый свободный (это не очевидно, я случайно пропустил комментарий). >>3) append - это вообще шедевр! >>функция добавляет элемент, но куда(позицию/индекс/ключ), она никому не >>скажет, потому что неположенно ;-) Если вы обратили внимание, то remove ломает всю индексную адресацию, т.к. при удалении элемента сжимает массив и толку от индексов в этом случае нет. Я пользуюсь другой адресацией элементов, не индексной. На самом деле класс опубликован не весь, только проблемные зоны, т.к. в других частях все нормально. Ну хоть посоветуйте какой-либо инструмент, вроде Valgrid только под FreeBSD... Я что-то не могу ничего найти, но ведь как-то FreeBSD'ишники отслуживают утечки и неправильные обращения с памятью.
- Освобождение памяти в деструкторе шаблонного класса, AnToXa, 20:14 , 05-Авг-05 (12)
>Если честно, то не знаю, в чем разница, т.к. не смотрел исходников >vector'a. Свой написал только потому, что не хочу пользоваться stdlib - >растет размер и падает скорость. это кто вам такую глупость сказал?
- Освобождение памяти в деструкторе шаблонного класса, xhk, 14:03 , 04-Авг-05 (7)
У вас тут банальная ошибочка допущена. Нельзя смешивать new/delete с malloc/free. В конструкторе выделение происходит с помощью malloc, а в remove память освобождается с помощью delete. Каша какая-то. И вообще realloc не самый лудший способ создания динамических массивов в плане производительности. Лучше выделять доп. куски памяти размером в страницу. Так куда эффективнее получитса.
- Освобождение памяти в деструкторе шаблонного класса, muxas, 15:34 , 04-Авг-05 (8)
Может быть вы правы? Я рассуждал так:В этой строке Malloc'ом выделяется место под массив указателей в конструкторе (AnyType==PointerToAnyObject): >> items = (AnyType*) malloc(sizeof(AnyType)*nstep); Это место зачищается с помощью строки в деструкторе: >> free(items); Сами объекты создаются где-то вне массива с помощбю new(). А строчка >> delete items[index]; возможно (это и есть первоначальный вопрос) будет удалять этот объект. Не массив указателей! Он удаляется в деструкторе. Т.е. весь вопрос в том, что делает эта последняя строка - удаляет она объект или нет. Вроде по логике, выражение items[index] должно возвращать указатель на объект из массива указателей. Следовательно теоретически его (этот объект) можно грохнуть изнутри массива оператором delete. Вот только что происходит на самом деле? Вот какой-нить С++'овый инструмент вроде Valgrind'а бы помог. Только ставить Linux ради него не хочется. Мог бы помочь к-нить гуру :-) тока его хрен найдешь. А может я где-то ошибаюсь в логике? У меня сильно смешана индексная и адресная арифметика, так что не все так просто, как кажется на первый раз. Возможно, следует перейти только к адресной.
- Освобождение памяти в деструкторе шаблонного класса, xhk, 19:29 , 04-Авг-05 (10)
>Сами объекты создаются где-то вне массива с помощбю new(). А строчка >>> delete items[index]; >возможно (это и есть первоначальный вопрос) будет удалять этот объект. Не массив >указателей! Он удаляется в деструкторе. Эта строка вообще не красива по сути. :) Вместо AnyType можно же подпихнуть и не класс, а можно вообще и не поинтер! В таком случае, что он вообще удалять то будет? Вы то, понятно, хотите туда пихать поинтер на класс, но самому классу DynArray это не известно. Тем более там нет проверки существования объекта по направлению самого поинтера.
- Освобождение памяти в деструкторе шаблонного класса, muxas, 15:58 , 04-Авг-05 (9)
И вообще realloc не самый лудший >способ создания динамических массивов в плане производительности. Лучше выделять доп. куски >памяти размером в страницу. Так куда эффективнее получитса. Насчет эффективности я так рассуждал - любой вызов new\malloc\etc порождает системный вызов. Только new() еще дополнительно выполняет некий код. Плюс еще такой момент - если памяти не хватает, то что делать если пользуешся new()? Делать delete и затем еще один new()? Или складывать массив из связанного списка кусочков памяти? Тогда всю эффективность адресации при одном куске памяти нахрен придется послать. Насчет кол-ва единовременно выделяемой памяти - сколько водки не бери все равно два раза бегать :-). Собственно говоря, страницы памяти под массив указателей возможно это очень много (смотря какая у вас страница). Размер указателя не очень большой. Однако это зависит от того для чего массивы использовать. Мне например нужно разместить результаты выборки из DВ. Там может быть и 5 строк и 100005. Так какой размер выбрать для единовременного выделения? Короче, я решил, что нужно задавать шаг выделения памяти при создании массива. Он задается в конструкторе, в массиве он держится в nstep. Не самый лучший способ, но зато гибкий :-). Есть идеи лучше? Поделитесь :-)
- Освобождение памяти в деструкторе шаблонного класса, xhk, 19:31 , 04-Авг-05 (11)
>Есть идеи лучше? Поделитесь :-) Просто realloc работает то не хитро. Он просто выделяет новую память, копирует туда все и удаляет старую. Это можно решить с помощью связанного списка из кусков памяти размером в страницу (для Linux размер страницы 4096, для Фрей - Гугль в помощь :) ). Когда в массиве будет нехватать памяти, просто выделяется дополнительно кусок памяти размером в страницу и указатель на нее пихается в список.
|