Главная страница
Top.Mail.Ru    Яндекс.Метрика
Текущий архив: 2005.06.29;
Скачать: CL | DM;

Вниз

Ошибка "Out of memory"   Найти похожие ветки 

 
TrueCoder   (2005-06-01 03:29) [0]

Так как Delphi пользуюсь нечасто, возникла проблема, решить которую сам пока не могу. Поиск тоже ничего что-то не дал..

Объявлен тип, например: type TValues = array [0..8] of Variant;
Объявлен массив:   Values: array of TValues;

Имеется счетчик элементов массива Values = Count: Integer.
Изначально массив пуст, значит Count=0.
Новые элементы массива Values добавляются в цикле по одному через SetLength(Values,Count+1). И все работает, до тех пор пока размер памяти, занимаемый приложением, не достигает 128 мегабайт. После чего, возникает ошибка "Out of memory".
Причем, размер оперативки в компе - 1 Гб, не говоря уже о виртуальной памяти. В момент возникновения ошибки оперативки свободно около 500 Мб. Выгрузка вообще всех других приложений тоже не помогает.

Вопрос - отчего возникает такая ошибка? Может я чего не понимаю в механизме выделения памяти? В хелпе имеется такая фраза по SetLength - "If there is not enough memory available to reallocate the variable, SetLength raises an EOutOfMemory exception", что у меня, похоже, и происходит. Но почему?!

ОС - WinXP Home, сама Delphi - семерка. Благодарю за любую помощь.


 
Defunct ©   (2005-06-01 04:35) [1]

Попробуйте заменить массив на TList, там SetLength используется немного не так как у вас.


 
Digitman ©   (2005-06-01 08:45) [2]

при очередном запросе Вorland Мemory Мanager"а на перераспределение памяти система не обнаружила в АП тек.процесса свободного региона затребованного размера и вернула отказ, о чем ВММ любезно и сообщил соотв.исключением

скорей всего АП процесса на момент отказа было сильно фрагментировано по причине предыдущих массированных "мелких" реаллокаций, инициированных вызовами SetLength()


 
ANB ©   (2005-06-01 08:57) [3]


> Digitman ©   (01.06.05 08:45) [2]
Имхо. Очень странно. Занято 128М. Дефрагментация, понятно, но у процесса 4Г адресного пространства. Как может не хватить ? Может проблема в стеке ?


 
Anatoly Podgoretsky ©   (2005-06-01 09:06) [4]

TrueCoder   (01.06.05 03:29)  
Где код?


 
Bel ©   (2005-06-01 09:09) [5]

> но у процесса 4Г адресного пространства

Причем, учти, из них половина отдана под ядро, а в оставшейся половине кроме данных твоего приложения располагаются exe-шник твоего приложения, а также все библиотеки. Плюс много других данных, используемых приложением. В итоге остается достаточно свободной памяти, но если расходовать ее неправильно, то и она может кончиться.


 
ANB ©   (2005-06-01 09:16) [6]


> Причем, учти, из них половина отдана под ядро
- Ну 2Г. Ну 500М накладные расходы (сроду столько не было). 1,5Г забить одним массивом ?
В принципе хороший совет - сначала определить максимальную длинну, а потом писать в массив, но он не всегда выполним. Например, я не всегда знаю заранее размер массива. В своем проекте у меня много мест, где длинна приращивается на 1, но на такую ошибку ни разу не нарывался. Ресурсы раз кончаться начали, но это я компонент кривой заюзал. Поменял на стандартный - ошибка ушла.


 
Bel ©   (2005-06-01 09:21) [7]

Дык, массив то должен занимать непрерывный участок памяти. Памяти то может быть и 1.5Г свободно, а вот непрерывного участка размером, допустим, 10М может не быть. Поэтому в данном случае уже предложили решение - использовать TList.


 
evvcom ©   (2005-06-01 09:22) [8]

Вообще подход действительно неверный. Реаллокэйтить такие массивы - это встретиться с жуткими тормозами из-за постоянного копирования "полезной" информации. Лучше использовать TList или цепочки.


 
Digitman ©   (2005-06-01 09:22) [9]


> ANB ©   (01.06.05 08:57) [3]


> Занято 128М


эта цифирь не имеет отношения к ВАП процесса


 
ЮЮ ©   (2005-06-01 09:32) [10]

Если человеку не нужны механизмы TList, то достаточно увеличивать длину массива так, как это сделано в TList.Grow, тем более, что количество действительных элементов массива он и так подсчитывает в Count


 
ANB ©   (2005-06-01 09:32) [11]


> эта цифирь не имеет отношения к ВАП процесса
- согласен. Но не полностью. Чем больше эта цифра, тем больше вероятности нарваться на конец памяти. Приплюсовать сюда дефрагментацию . . . Кстати, автор не дал код и не сообщил максимальную длинну своего массива.


 
Mx ©   (2005-06-01 09:46) [12]

Возник вопрос: а чем же TList лучше? В не тоже есть Realloc"и, хоть и реже вызываемые, но в итоге точно также потребуется большой объем памяти (правда Pointer меньше чем Variant, но это лишь вопрос количества элементов).


 
Думкин ©   (2005-06-01 09:48) [13]

> [12] Mx ©   (01.06.05 09:46)

хоть и реже вызываемые, - ты сказал(с)


 
Mx ©   (2005-06-01 09:55) [14]


> Думкин ©   (01.06.05 09:48) [13]

Вроде ошибка: Out of memory, а не Unable to Realloc? Не хватает, как я понял, объема, причем здесь количество реалокейтов?


 
ANB ©   (2005-06-01 09:56) [15]

Без кода и поллитры не разберемся.


 
Думкин ©   (2005-06-01 10:05) [16]

> [14] Mx ©   (01.06.05 09:55)

смотрим [2] Digitman ©   (01.06.05 08:45) - задумываемся.


 
Mx ©   (2005-06-01 10:13) [17]


> Думкин ©   (01.06.05 10:05) [16]

Я уже читал. Но: Дык, массив то должен занимать непрерывный участок памяти. Памяти то может быть и 1.5Г свободно, а вот непрерывного участка размером, допустим, 10М может не быть. Поэтому в данном случае уже предложили решение - использовать TList. © Bel. Ну допустим, Realloc не смог выделить память в текущем месте, я так понимаю занимаемую память он освободит, и будет выделен новый - бОльший участок в другом месте. Ну так и причем здесь количество Realloc"ов? Или он работает только "по возрастанию"? Т.е. в конце концов может "упереться в вершину" (извините, за образность)?


 
evvcom ©   (2005-06-01 10:17) [18]


> Mx ©   (01.06.05 10:13) [17]

Каков характер твоих данных? Необходим ли тебе доступ к произвольному элементу этого массива в любой момент времени? Или эти данные обрабатываются строго по порядку (сверху вниз или снизу вверх, не важно)?


 
Mx ©   (2005-06-01 10:20) [19]

У меня данных нет, почитал ветку и кое чего не понял, вот и спросил... вдогонку.


 
evvcom ©   (2005-06-01 10:24) [20]

Да, действительно... [18] должно быть адресовано TrueCoder


 
Mx ©   (2005-06-01 10:31) [21]

Продолжаю. Допустим у нас 20 ячеек памяти, каждая пятая занята. Тогда какая разница, что я "дойду" до необходимости выделения 6 смежных ячеек, а что я их сразу запрошу? Свободных то все-равно нет. Это я имел ввиду в вопросе.


 
Digitman ©   (2005-06-01 10:33) [22]


> Mx ©   (01.06.05 09:55) [14]
> Вроде ошибка: Out of memory, а не Unable to Realloc? Не
> хватает, как я понял, объема, причем здесь количество реалокейтов?


найти грабли достаточно легко - перехватываем точки входа в VirtualAlloc и LocalAlloc, протоколируем отказы этих вызовов

если отказал VirtualAlloc, грабли в дефрагментации и/или отсутствии региона нужного размера

если LocalAlloc, грабли - в исчерпании кучи


 
Mx ©   (2005-06-01 10:38) [23]

У меня примера нет. Может автор проверит?


 
Digitman ©   (2005-06-01 10:41) [24]

getmem.inc наглядно показывает grow-механизм работы ВММ - сначала в АП резервируется регион, а затем идет запрос к дифолтной куче

а в целом грабли даже и искать не нужно

лезем в sysutils.pas, видим там

EOutOfMemory = class(EHeapException);

т.е. отказ вернула ф-ция LocalAlloc из-за исчерпания кучи


 
Mx ©   (2005-06-01 10:43) [25]

Как на это влияет число вызовов ReallocMem?


 
Digitman ©   (2005-06-01 11:00) [26]

вот так и влияет

AddBlockAfter() вызывает LocalAlloc() посредством GetBlockDesc(), запрашивая из кучи блок памяти, а DeleteBlock() назад в кучу его не отдает


 
Mx ©   (2005-06-01 11:32) [27]

Все-равно не понимаю. Если я запрошу блок вначале 1*Integer, потом 2*Integer, потом 3*Integer. А если запрошу сразу 3*Integer, то что выиграю? И в том и в другом случае я запрашиваю 3*Integer ячеек. А! В первом случае получается 6*Integer, да? Правильно понял? Т.е., высвобождая предыдущий блок, ReallocMem не далает эти ячейки доступными для следующего Realloc"а?


 
Digitman ©   (2005-06-01 11:58) [28]


> Mx ©   (01.06.05 11:32) [27]


> Если я запрошу блок вначале 1*Integer, потом 2*Integer,
> потом 3*Integer.


.. то потенциально возможны три вызова LocalAlloc() при ни одном LocalFree()


А если запрошу сразу 3*Integer, то что
> выиграю?


при этом будет не более чем один LocalAlloc()

ВММ оперирует понятием и механизмом "блок"

у каждого блока есть упр.структура для организации связного двунапр.списка из блоков, хранения атрибутов ("свободен", "занят"), адреса, где размещены собственно данные блока, если блок занят, и размера.

когда прикл.код обращается к ВММ за выделением памяти, ВММ ищет первый своб.блок минимально подходящего размера и помечает его как "занятый", манипулируя далее с соотв.регионами в АП процесса (1)

если своб.блок не найден, ВММ запрашивает из кучи фрагмент размера равного заголовку блока и затем выполняет (1)

когда прикл.код обращается к ВММ за освобождением памяти, ВММ ищет заголовок блока по полю размера, помечает этот блок как свободный, НЕ отдавая в кучу память, ранее запрошенную под этот блок, и далее опять же манипулирует соотв.регионами

реаллокация у ВММ - более сложный алгоритм (с т.з. манипуляций регионами), но сводится он опять же к аллокации нового и деаллокации прежнего блока

ну это так, сильно упрощенная схема ...
детали же реально происходящего - в getmem.inc


 
Mx ©   (2005-06-01 12:35) [29]

Как все запушено... Спасибо, на досуге повнимательнее почитаю getmem.inc.


 
TrueCoder   (2005-06-01 16:38) [30]

Большое спасибо всем ответившим. Даже не ожидал, что моя проблема поднимет такую дискуссию. Есть о чем подумать.
Трезвую мысль высказал Digitman: "Cкорей всего АП процесса на момент отказа было сильно фрагментировано по причине предыдущих массированных мелких реаллокаций, инициированных вызовами SetLength()". Интуитивно чувствую, что истина где-то тут.
Код привести, к сожалению, не могу, запутанный он. Там долго и нудно обрабатываются Variant-ячейки таблицы ExpressQuantumGrid (штука классная, но сложнаяяяя..). Исключение возникает при обработке порции порядка 5 тысяч записей, но записи большие, в каждой по 50 полей, большинство текстовые, разной длины, поэтому я даже затруднюсь сказать максимальный размер массива.

Главное, что я понял - ошибка где-то в самом коде. Точнее, даже не в коде, а в самом принципе использования памяти - не любит ОНО многочисленных мелких реалокаций. Буду думать, может TList прикрутить получится.

Насторожило то, что исключение возникает при объеме памяти, занимаемого приложением, в 128 Мб. По неопытности и подумал, может по умолчанию при компиляции какие ограничения существуют. А нет, чуда не произошло. :)


 
Digitman ©   (2005-06-01 17:05) [31]


> TrueCoder   (01.06.05 16:38) [30]


> Трезвую мысль высказал Digitman


да ну нет)... на более пристальный взгляд она не такая уж и трезвая оказалась ..

EOutOfMemory = class(EHeapException);

т.е. отказ этот вызван исчерпанием хипа, а с хипом работает LocalAlloc, а LocalAlloc вызывается для аллокации памяти под заголовок блока, а не под сами данные, на которые занятый блок ссылается

отсюда и вывод напрашивается - с дефрагментацией АП еще бабушка надвое сказала, а вот хип опустошен тобой множеством реаллокаций по самое нехочу


> Буду думать, может TList прикрутить получится


ТЛист пользует тот же менеджер памяти, разница лишь в grow-алгоритме : ТЛист при добавлении эл-та списка запрашивает память сразу под несколько (не помню сколько - см. код класса) элементов, т.о. минимизируя число вновь создаваемых блоков и, соответственно, новых аллокациий в хипе

рано или поздно и с ТЛист можно нарваться на ту же ситуацию


> Насторожило то, что исключение возникает при объеме памяти,
> занимаемого приложением, в 128 Мб


а ты не гадай !
ты проверь или отвергни предположение насчет хипа ..

сразу после старта приложения запроси у системы хэндл дифолтного хипа своего процесса (GetProcessHeap) и тут же просканируй хип (HeapWalk)

а потом перед каждой своей итерацией, где ты реаллокируешь свой массив, делай тоже самое и следи за тем как изменяется состояние хипа


 
Anatoly Podgoretsky ©   (2005-06-01 20:40) [32]

Digitman ©   (01.06.05 17:05) [31]
ТЛист пользует тот же менеджер памяти, разница лишь в grow-алгоритме : ТЛист при добавлении эл-та списка запрашивает память сразу под несколько (не помню сколько - см. код класса) элементов, т.о. минимизируя число вновь создаваемых блоков и, соответственно, новых аллокациий в хипе

И не запомнишь, у него адаптирующий алгоритм, с переменным количеством сколько. Вроде бы каждое увеличение на 25% от текущего размера блока, чем больше запрашиваешь, чем больше получаешь.


 
Defunct ©   (2005-06-02 01:21) [33]

Anatoly Podgoretsky ©   (01.06.05 20:40) [32]
> разница лишь в grow-алгоритме

Не только. TList также был предложен по той причине, что размер непрерывного участка памяти при использовании TList будет минимально возможным, без переделывания логики программы.


 
TrueCoder   (2005-06-03 23:39) [34]

Наверное, никому уже не интересно, но все же расскажу. Анализом кода удалось найти возможность заранее вычислять размер будущего массива, поэтому число реалокаций с 5 тысяч удалось снизить до двух. И проблема отпала сама собой.
Конечно, текущее решение значительно более красивое, как с точки зрения чистого кодинга, так и с точки зрения полученного удовлетворения от красивого решения проблемы :), но, все же, осталось некоторое удивление. Разве, стабильно работающая система (я о менеджере памяти Delphi и о самой операционке) не должны стабильно работать и при условии пусть некрасивых, но все же верных решений. Таких как 5 тысяч мелких реалокаций памяти у меня ранее. Что-то все же неладно в данном королевстве..
Интерес остался, конечно, исключительно чисто теоретический, проверять почему не работало раньше банально нет времени.


 
Defunct ©   (2005-06-04 02:53) [35]

> TrueCoder

Проблема фрагментации - вечная проблема. Менеджер памяти пойдет в ущер скорости исполения, если будет заниматься постоянной дефрагментацией.
Поэтому делайте всегда всё красиво.



Страницы: 1 вся ветка

Текущий архив: 2005.06.29;
Скачать: CL | DM;

Наверх




Память: 0.57 MB
Время: 0.032 c
1-1118169137
Ralf
2005-06-07 22:32
2005.06.29
Form


6-1112367124
BackGround
2005-04-01 18:52
2005.06.29
Socket.Data


1-1118254543
нехочу называться
2005-06-08 22:15
2005.06.29
Упростить конструкцию


3-1116339331
MEV
2005-05-17 18:15
2005.06.29
Создать базу данных runtime через ADO


3-1116568506
Kirpenko
2005-05-20 09:55
2005.06.29
ER-Win