Форум программистов, компьютерный форум CyberForum.ru

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
VladK
0 / 0 / 0
Регистрация: 13.05.2014
Сообщений: 26
#1

Динамические массивы: выделяем и высвобождаем память - C++

25.08.2014, 19:10. Просмотров 856. Ответов 21
Метки нет (Все метки)

Доброго времени суток!
Динамические массивы: выделяем и высвобождаем память. Но сложность в том, что после выполнения delete[], память полностью не высвобождается. Когда в течении процесса, приходится выделять и высвобождать довольно много раз, то память растет как снежный ком.
Как возможно решить данную проблему?

Для теста:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
// MemTest.cpp : Defines the entry point for the console application.
//
#include "stdafx.h"
#include <iostream>
#include <windows.h>
#include <psapi.h>
#pragma comment(lib, "psapi.lib") 
 
using namespace std;
 
void GetMemSize(DWORD ProcessID, size_t &Memory, size_t &PageFile)
{
    HANDLE hProcess;
    PROCESS_MEMORY_COUNTERS pmc;
 
    hProcess = OpenProcess(  PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, ProcessID );
    if (NULL == hProcess)
        return ;
 
    if (GetProcessMemoryInfo( hProcess, &pmc, sizeof(pmc)))
    {
        Memory = pmc.WorkingSetSize;
        PageFile = pmc.PagefileUsage;
    }
    CloseHandle( hProcess );
}
 
void SingleArr()
{
    int *arr = new int [100500];
    delete []arr;
}
 
void MultiArr()
{
    int nRow = 10, i;
    int **multiarr = new int *[nRow];
    for(i = 0; i < nRow; i++)
        multiarr[i] = new int [100500];
 
    for(i = 0; i < nRow; i++)
        delete []multiarr[i];
    delete []multiarr;
}
 
int _tmain(int argc, _TCHAR* argv[])
{
    DWORD processID = GetCurrentProcessId();
    size_t memory_before, pagefile_beforre, memory_after, pagefile_after; 
 
    GetMemSize(processID, memory_before, pagefile_beforre);
 
    SingleArr();
 
    GetMemSize(processID, memory_after, pagefile_after);
    cout << "Before\tWorkingSetSize: " << memory_before << "\tPagefileUsage: " << pagefile_beforre << endl;
    cout << "After\tWorkingSetSize: " << memory_after << "\tPagefileUsage: " << pagefile_after << endl << endl;
    cout << "Used\tWorkingSetSize: " << memory_after - memory_before << "\tPagefileUsage: " << pagefile_after - pagefile_beforre << endl << endl;
 
    GetMemSize(processID, memory_before, pagefile_beforre);
 
    MultiArr();
 
    GetMemSize(processID, memory_after, pagefile_after);
    cout << "Before\tWorkingSetSize: " << memory_before << "\tPagefileUsage: " << pagefile_beforre << endl;
    cout << "After\tWorkingSetSize: " << memory_after << "\tPagefileUsage: " << pagefile_after << endl << endl;
    cout << "Used\tWorkingSetSize: " << memory_after - memory_before << "\tPagefileUsage: " << pagefile_after - pagefile_beforre << endl << endl;
 
    system ("pause");
    return 0;
}
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
25.08.2014, 19:10
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Динамические массивы: выделяем и высвобождаем память (C++):

Динамические массивы: очищается ли память? - C++
Здравствуйте. Подскажите, пожалуйста, по поводу динамических массивов. Допустим, есть вот такой код: int *massive = new int; int...

Структуры: зачем мы выделяем память сами, когда это можно сделать автоматически? - C++
есть кусок программы: struct controller { struct watching_area* wa; // 1 int skill; char* name; }; struct...

Динамические массивы и функции. Динамические массивы. - C++
Дана матрица 6х8 целого типа. Создать одномерный массив, содержащий элементы матрицы, кратные числу, введенному с клавиатуры.

Указатели и массивы. Индексация с помощью указателей. Передача массивов в функции. Динамические массивы (обработка матриц) - C++
Для каждого элемента , bij, i= 1,...,n , j=1,...,n определяется свой многоугольник местонахождением соответствующего элемента aij (см....

Как выделить память под динамические переменные при создании класса? - C++
В прошлой теме ответ получил, но не до конца. Вопрос такой, как выделить память под динамические переменные? Вот часть кода ...

Динамические массивы - C++
помогите выполнить задание #include &lt;stdio.h&gt; #include &lt;stdlib.h&gt; #include &lt;math.h&gt; #include &lt;time.h&gt; void main() { int...

Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
VladK
0 / 0 / 0
Регистрация: 13.05.2014
Сообщений: 26
26.08.2014, 00:36  [ТС] #16
Цитата Сообщение от DrOffset Посмотреть сообщение
А это и не проблема.
Если без подробностей, то не верь таск менеджеру.
А подробности тут.
По этому добавил чтение памяти до и после процесса

Добавлено через 11 минут
Цитата Сообщение от КОП Посмотреть сообщение
Странно, добавил свою функцию
Код C++
void SomeFunc()
{
int* i;
i = new int[100500];
delete[] i;
}
Вставил точно так же в код, после нее потери 0, а ненулевые только после MultiArr() и SingleArr()
Значит правильно люди говорят, особенности управления памятью со стороны системы.
Не знаю как получается 0 !? в таск менеджере или после чтения памяти процесса?
КОП
361 / 280 / 86
Регистрация: 15.08.2010
Сообщений: 762
26.08.2014, 00:54 #17
Цитата Сообщение от VladK Посмотреть сообщение
Не знаю как получается 0
Полагаю, ответ уже дан. Система чудит с памятью.

Но для полноты картины:
Кликните здесь для просмотра всего текста

http://www.cyberforum.ru/attachment....1&d=1408999994

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
void GetMemSize(DWORD ProcessID, size_t &Memory, size_t &PageFile)
{
    HANDLE hProcess;
    PROCESS_MEMORY_COUNTERS pmc;
 
    hProcess = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, ProcessID);
    if (NULL == hProcess)
        return;
 
    if (GetProcessMemoryInfo(hProcess, &pmc, sizeof(pmc)))
    {
        Memory = pmc.WorkingSetSize;
        PageFile = pmc.PagefileUsage;
    }
    CloseHandle(hProcess);
}
 
void SingleArr()
{
    int *arr = new int[100500];
    delete[]arr;
}
 
void MultiArr()
{
    int nRow = 10, i;
    int **multiarr = new int *[nRow];
    for (i = 0; i < nRow; i++)
        multiarr[i] = new int[100500];
 
    for (i = 0; i < nRow; i++)
        delete[]multiarr[i];
    delete[]multiarr;
}
 
void SomeFunc()
{
    int* i;
    i = new int[100500];
    delete[] i;
}
 
int _tmain(int argc, _TCHAR* argv[])
{
    DWORD processID = GetCurrentProcessId();
    size_t memory_before, pagefile_beforre, memory_after, pagefile_after;
 
    GetMemSize(processID, memory_before, pagefile_beforre);
 
    SingleArr();
 
    GetMemSize(processID, memory_after, pagefile_after);
    cout << "Before\tWorkingSetSize: " << memory_before << "\tPagefileUsage: " << pagefile_beforre << endl;
    cout << "After\tWorkingSetSize: " << memory_after << "\tPagefileUsage: " << pagefile_after << endl << endl;
    cout << "Used\tWorkingSetSize: " << memory_after - memory_before << "\tPagefileUsage: " << pagefile_after - pagefile_beforre << endl << endl;
 
    GetMemSize(processID, memory_before, pagefile_beforre);
 
    SomeFunc();
 
    GetMemSize(processID, memory_after, pagefile_after);
    cout << "Before\tWorkingSetSize: " << memory_before << "\tPagefileUsage: " << pagefile_beforre << endl;
    cout << "After\tWorkingSetSize: " << memory_after << "\tPagefileUsage: " << pagefile_after << endl << endl;
    cout << "Used\tWorkingSetSize: " << memory_after - memory_before << "\tPagefileUsage: " << pagefile_after - pagefile_beforre << endl << endl;
 
    GetMemSize(processID, memory_before, pagefile_beforre);
 
    MultiArr();
 
    GetMemSize(processID, memory_after, pagefile_after);
    cout << "Before\tWorkingSetSize: " << memory_before << "\tPagefileUsage: " << pagefile_beforre << endl;
    cout << "After\tWorkingSetSize: " << memory_after << "\tPagefileUsage: " << pagefile_after << endl << endl;
    cout << "Used\tWorkingSetSize: " << memory_after - memory_before << "\tPagefileUsage: " << pagefile_after - pagefile_beforre << endl << endl;
 
    system("pause");
    return 0;
}
Миниатюры
Динамические массивы: выделяем и высвобождаем память  
DrOffset
7092 / 4233 / 950
Регистрация: 30.01.2014
Сообщений: 7,008
26.08.2014, 01:03 #18
Сборка релиз или дебаг? Если дебаг, то там еще хранится информация о выделенной памяти, для того, чтобы можно было отследить утечки. И это может "путать карты" при замере. Ссылку читал?

Цитата Сообщение от VladK Посмотреть сообщение
По этому добавил чтение памяти до и после процесса
Ага понятно.
Для такого применения WorkingSetSize не подходит. Нужно использовать PROCESS_MEMORY_COUNTERS_EX и поле PrivateUsage. По документации это тоже самое, что PagefileUsage, но если у тебя win7, то PagefileUsage будет всегда равен нулю, поэтому смотреть надо PrivateUsage - значение показывает общее количество памяти, которое выделил менеджер памяти для данного процесса в байтах.

Вот еще тут почитай. Лучше все равно сложно написать.

Короче нет проблемы тут никакой.
VladK
0 / 0 / 0
Регистрация: 13.05.2014
Сообщений: 26
26.08.2014, 01:05  [ТС] #19
Цитата Сообщение от ValeryS Посмотреть сообщение
кто это сказал?
здесь возможно вторая засада сегментирование памяти
примерно так
есть большой кусок памяти выделили памяти 10 байт
потом 20
потом 10 байт удалили
потом выделили 30 байт
в первый кусок, свободный где 10 байт, эти 30 не влезут значит выделится новая память
а если учесть что менеджер выделяет память не побайтно а страницами, размер страницы может быть разный, зависит от системы
вывод если нужно выделять/удалять нужно писать свой менеджер
Возможно так и происходит, но с другой стороны если выделил Х то и высвободил Х, а не Х -/+ Y
ValeryS
Модератор
6556 / 5022 / 464
Регистрация: 14.02.2011
Сообщений: 16,763
26.08.2014, 01:12 #20
Цитата Сообщение от VladK Посмотреть сообщение
но с другой стороны если выделил Х то и высвободил Х, а не Х -/+ Y
Это уж как менеджер решит
он вполне может не удалять при каждом delete а чохом освободить сразу кучу( в смысле много, а не в смысле организации памяти)
вот ты выделяешь/ удаляешь выделяешь/ удаляешь а у него еще куча памяти он на использованную ставит метку а выделяет новую, а потом когда не хватит памяти чохом всю и очистит
Выделение/ удаление дело очень дорогое и посему пытается оптимизировать
да и Дебиг режим мог свое слово сказать о чем говорил DrOffset,
VladK
0 / 0 / 0
Регистрация: 13.05.2014
Сообщений: 26
26.08.2014, 02:22  [ТС] #21
Цитата Сообщение от КОП Посмотреть сообщение
Полагаю, ответ уже дан. Система чудит с памятью.
Согласен, чудит
Интересно то, что если в Вашем варианте сначало выполнить SomeFunc(); а потом SingleArr(); , то результат
второй функции будет 0.
Решил добить , добавил четвертый вызов MultiArr(); , но замерял память толко в начале и после четвертого вызова, так высвободилось больше, чем выделялось!
Спасибо, за определение !!!

Добавлено через 26 минут
Цитата Сообщение от DrOffset Посмотреть сообщение
Сборка релиз или дебаг? Если дебаг, то там еще хранится информация о выделенной памяти, для того, чтобы можно было отследить утечки. И это может "путать карты" при замере. Ссылку читал?
Цитата Сообщение от VladK Посмотреть сообщение
По этому добавил чтение памяти до и после процесса
Ага понятно.
Для такого применения WorkingSetSize не подходит. Нужно использовать PROCESS_MEMORY_COUNTERS_EX и поле PrivateUsage. По документации это тоже самое, что PagefileUsage, но если у тебя win7, то PagefileUsage будет всегда равен нулю, поэтому смотреть надо PrivateUsage - значение показывает общее количество памяти, которое выделил менеджер памяти для данного процесса в байтах.
Вот еще тут почитай. Лучше все равно сложно написать.
Короче нет проблемы тут никакой.
Спасибо за линк !!! Возможно проблемы нет, но при сборке релиз после запуска программа память увеличевается с 30М до 400М. Утечки памяти нет, проверенно. Осталось проверять выделение и высвобождение памяти

Добавлено через 24 минуты
Цитата Сообщение от ValeryS Посмотреть сообщение
Это уж как менеджер решит
он вполне может не удалять при каждом delete а чохом освободить сразу кучу( в смысле много, а не в смысле организации памяти)
вот ты выделяешь/ удаляешь выделяешь/ удаляешь а у него еще куча памяти он на использованную ставит метку а выделяет новую, а потом когда не хватит памяти чохом всю и очистит
Выделение/ удаление дело очень дорогое и посему пытается оптимизировать
да и Дебиг режим мог свое слово сказать о чем говорил DrOffset,
Не уверен что чохом, ведь new/delete это конструктор и деструктор. Как я понимаю, деструктор затерает начальный адрес объекта, освобождая память.
ValeryS
Модератор
6556 / 5022 / 464
Регистрация: 14.02.2011
Сообщений: 16,763
26.08.2014, 08:55 #22
Цитата Сообщение от VladK Посмотреть сообщение
Не уверен что чохом, ведь new/delete это конструктор и деструктор.
есть функция new, а есть оператор
не они выделяют память а система, точнее менеджер памяти, они только запрашивают у системы память
чаще всего реализуется через malloc который уже и разговаривают с системой
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
26.08.2014, 08:55
Привет! Вот еще темы с ответами:

Динамические массивы - C++
Суть задания такова. Создается динамический массив А, заполняется элементами. Затем создается массив B, куда записываются все элементы...

Динамические массивы - C++
Добрый вечер! Проблема такая,плохо понимаю,как изменить в моей программе обычные массивы на динамические. Надеюсь,кто-то поможет и...

Динамические массивы - C++
Объясните пожалуйста что такое динамические массивы. Это массивы с не константным количеством элементов? Если так, то как их объявлять?

Динамические массивы - C++
Подскажите, как найти в одномерном динамическом массиве номер первого элемента, на котором произошла смена знака?(язык Си)


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
Yandex
Объявления
26.08.2014, 08:55
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru