Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.61/114: Рейтинг темы: голосов - 114, средняя оценка - 4.61
Заблокирован

Когда вызывается деструктор класса?

07.10.2012, 11:56. Показов 22912. Ответов 22
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Вопрос собственно в том, когда происходит освобождение памяти, занимаемой объектом класса? Допустим имеем
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <iostream>
struct danger{
    danger(int* intermedval):d(intermedval){};
    int get_d(){return *d;}
    private:
    int* d;
    ~danger(){delete d;}
    };
 
int main(){
    int* i =new int (1050);
    danger dang(i);
    delete i;
    std::cout<<dang.get_d()<<std::endl;
 
 
{огромная куча кода}
 
 
    return 0;
    }
получается что объект освободит память, когда сработает return в main?
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
07.10.2012, 11:56
Ответы с готовыми решениями:

При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О
Вот такой кодclass A { public: A(){} virtual ~A(){} }; class C { public:

Вызывается деструктор когда обьект не уничтожен!
#include &quot;stdafx.h&quot; #include &lt;iostream&gt; #include &lt;conio.h&gt; using namespace std; class inkr { private: int a; ...

Будет ли утрачена память, когда конструктор копирования вызывается для уже существующего объекта класса?
class A { char * v; A(); ~A(); A(const A &amp;obj); } ///////////////////// A::A() {

22
 Аватар для Кот Ангенс
320 / 270 / 128
Регистрация: 24.05.2012
Сообщений: 629
07.10.2012, 12:22
AnreyKazakov, по-первых, почему у Вас деструктор приватный? Во-вторых, почему Вы удаляете число до завершения работы с объектом?
Цитата Сообщение от AnreyKazakov Посмотреть сообщение
получается, что объект освободит память, когда сработает return в main?
Да. Но если надо раньше, можно явно указать область видиости:
C++
1
2
3
4
5
6
7
8
9
10
int main() {
    int* i = new int(100500);
    {
        danger dang(i);
        std::cout << dang.get_d() << std::endl;
        //dang.~danger();
    }
    delete i;
    //Огромная куча кода
}
1
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
07.10.2012, 19:57
AnreyKazakov, деструктор автоматически вызывается компилятором в тот момент, когда объект выходит из области видимости. Пример:
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
#include <iostream>
 
class Foo
{
public:
    ~Foo()
    {
        std::cout << "Foo::~Foo()" << std::endl;
    }
};
 
int main()
// first scope start
{
    Foo foo1; // creating first instance
    
    // second scope start
    {
        Foo foo2; // creating second instance
    // second scope end; all objects with this scope (viz. foo2) are destroyed
    }
    
    std::cout << "Hello, World!" << std::endl; // for visual separation of destructors output
    
    return 0;
// first scope end; all objects with this scope (viz. foo1) are destroyed
}
http://liveworkspace.org/code/... 36d6638897
2
5499 / 4894 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
08.10.2012, 01:23
Цитата Сообщение от AnreyKazakov Посмотреть сообщение
когда происходит освобождение памяти, занимаемой объектом класса?
Так же как и для других переменных. В зависимости от того, где этот объект создавался.
1
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
08.10.2012, 01:29
Цитата Сообщение от alsav22 Посмотреть сообщение
где этот объект создавался
Где и как, я бы сказал))
0
5499 / 4894 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
08.10.2012, 01:50
Цитата Сообщение от Кот Ангенс Посмотреть сообщение
Да. Но если надо раньше, можно явно указать область видиости:
Так будет ошибка. Повторное освобождение памяти, на которую i указывает.

Добавлено через 17 минут
Не пойму только, почему у AnreyKazakov в коде эта ошибка не возникает? Ведь там, при выходе из main(), будет вызван деструктор, и тоже будет попытка освобождения памяти, котрая уже была освобождена.
1
Заблокирован
08.10.2012, 04:18  [ТС]
Цитата Сообщение от alsav22 Посмотреть сообщение
Не пойму только, почему у AnreyKazakov в коде эта ошибка не возникает? Ведь там, при выходе из main(), будет вызван деструктор, и тоже будет попытка освобождения памяти, котрая уже была освобождена.
А я не пойму, что за ошибка должна выползти, все работает, только деструктор в public перекинул, пашет на все 100....
0
5499 / 4894 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
08.10.2012, 04:20
А с таким кодом работает?
C++
1
2
3
4
5
6
7
8
9
int main() 
{
    int* i = new int(100500);
    {
        danger dang(i);
    }
    delete i;
    //Огромная куча кода
}
0
 Аватар для Кот Ангенс
320 / 270 / 128
Регистрация: 24.05.2012
Сообщений: 629
08.10.2012, 07:39
Цитата Сообщение от alsav22 Посмотреть сообщение
Так будет ошибка. Повторное освобождение памяти, на которую i указывает.
Упс, не заметил, что в деструкторе она тоже освобождается.
Но здесь нарушается логика: подпрограмма не должна освобождать то, что она сама не выделяет.
0
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
08.10.2012, 07:48
Цитата Сообщение от Кот Ангенс Посмотреть сообщение
Но здесь нарушается логика: подпрограмма не должна освобождать то, что она сама не выделяет.
Зависит от ситуации и от стратегии владения. Например, все смартпоинтеры не выделяют память сами, но все её освобождают. Просто когда программист отдаёт выделенную память в смартпоинтер, он соглашается с тем, что теперь этой памятью смартпоинтер будет владеть безраздельно.
0
5499 / 4894 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
08.10.2012, 07:51
Можете что-нибудь сказать по моему вопросу в 6 посте?
0
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
08.10.2012, 07:59
alsav22, могу сказать только, что проблема скорее всего в компиляторе. Ибо если он ничего не сказал по поводу того, что деструктор имеет модификатор доступа privat, то это плохой компилятор. Дальше можно погрузиться в область догадок. Например, компилятор, увидев, что деструктор приватный, тупо не стал его вызывать. У меня после переноса деструктора в публичную область (до этого код не компилировался) вываливается кордамп по причине double free, как и должно быть.

Добавлено через 1 минуту
А, ну там была оговорка о том, что деструктор перекинут в паблик... Тогда я развожу руками.

AnreyKazakov, покажите, какой вид принял код теперь (который у вас работает).
1
5499 / 4894 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
08.10.2012, 08:09
Вот этот работает без ошибок (mingv)
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
#include <iostream>
#include <cstdlib>
using namespace std;
 
struct danger
{
    public:
      danger(int* intermedval):d(intermedval){};
 
      int get_d(){return *d;}
 
       ~danger()
       {
          cout << "d" << endl;
          delete d;
       }
    private:
       int* d;
 };
 
int main()
{
    int* i = new int (1050);
    danger dang(i);
    delete i;
 
 
    system ("pause");
    return 0;
}
Миниатюры
Когда вызывается деструктор класса?  
0
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
08.10.2012, 08:17
alsav22, оччччень странно... Я на работе сейчас, GCC 4.4.5, Ubuntu 10.04, кордамп. Приду домой, попробую на 4.7.1 на убунте и на мингв.
0
5499 / 4894 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
08.10.2012, 10:03
В Студии странно. Когда вопрос задавал, этот код в Студии отрабатывал без ошибок, сейчас (после срабатывания деструктора) виснет.

Добавлено через 4 минуты
В Dev-Cpp, без ошибок. Там тоже mingv, но другой.

Добавлено через 1 час 26 минут
В Linux, GNU GCC - free double.
0
 Аватар для taras atavin
4226 / 1796 / 211
Регистрация: 24.11.2009
Сообщений: 27,562
08.10.2012, 10:17
Деструктор вызывается ровно в двух случаях:
1. При выходе автоматического объекта за видимость.
2. При удалении динамического объекта делитом.
0
Заблокирован
08.10.2012, 12:16  [ТС]
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <iostream>
struct danger{
    danger(int* intermedval):d(intermedval){};
    int get_d(){return *d;}
    ~danger(){delete d;}
    private:
    int* d;
    };
 
int main(){
    int* i =new int (1050);
    danger dang(i);
    delete i;
    std::cout<<dang.get_d()<<std::endl;
 
 
//{огромная куча кода}
 
 
    return 0;
    }
MinGW компиь все работает, выдает 0.
Тыкните мне пальцем, где я не правильно сделал...

Добавлено через 2 минуты
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <iostream>
struct danger{
    danger(int* intermedval):d(intermedval){};
    int get_d(){return *d;}
    ~danger(){delete d;}
    private:
    int* d;
    };
 
int main(){
    int* i =new int (1050);
    danger dang(i);
    std::cout<<dang.get_d()<<std::endl;
    delete i;
 
 
//{огромная куча кода}
 
 
    return 0;
    }
Выдает 1050
0
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
08.10.2012, 12:32
AnreyKazakov, смотрите, где неправильно: в строке 12 вы размещаете в динамической памяти переменную типа int. В строке 13 вы передаёте адрес выделенной в предыдущей строке области памяти в конструктор класса danger; конструктор сохраняет этот адрес у себя во внутреннем указателе. В строке 15 вы явным образом освобождаете память, выделенную в строке 12. Она больше не принадлежит вашей программе, вы больше не имеете права как-либо её использовать. В строке 21 объект dang класса danger выходит из области своей видимости, в связи с чем вызывается его деструктор. Бэмс! В строке 6, в теле деструктора, вы повторно освобождаете память, которую выделили в строке 12 и уже освободили в строке 15. Что и должно вызвать проблему (и вызывает у меня на убунте).
1
Заблокирован
08.10.2012, 12:49  [ТС]
А как же
Пока жив хоть один указатель, жив и объект.
0
Эксперт С++
5058 / 3118 / 271
Регистрация: 11.11.2009
Сообщений: 7,044
08.10.2012, 12:53
AnreyKazakov, а при чём здесь нативные указатели? Это справедливо для т.н. умных указателей, который используют различные алгоритмы для отслеживания "живости" объектов. Самый простой такой алгоритм - подсчёт ссылок. Как только счётчик обнуляется - объект физически освобождается. А нативные указатели к этому не относятся - выделили вы памяти, храните хоть сотню указателей на неё, но освободив эту память с помощью хотя бы одного указателя, вы потеряете её навсегда.
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
08.10.2012, 12:53
Помогаю со студенческими работами здесь

Деструктор не вызывается
В одном классе я создаю объект, и если проверку не проходит, я его возвращаю в другой метод, дабы у меня там начало всех начал,Главное...

Не вызывается деструктор
#include &lt;iostream&gt; using namespace std; class myclass { int a; public: myclass(); // конструктор ~myclass(); //...

Не вызывается деструктор
_Здравствуйте. Я новичок в программировании, сейчас изучаю самостоятельно С++ по книге Джесса Либерти. При разборе одного из примеров...

Почему вызывается деструктор?
Доброго времени суток. Написал такой код: struct O { virtual ~O() { cout &lt;&lt; &quot;hey&quot;; }...

Почему не вызывается деструктор?
Всем доброго времени суток! Подскажите пожалуйста - почему не вызывается деструктор при выполнении строки 48? ...


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

Или воспользуйтесь поиском по форуму:
20
Ответ Создать тему
Новые блоги и статьи
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка.
Programma_Boinc 23.12.2025
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка. Рецензия / Мнение Это мой обзор планшета X220 с точки зрения школьника. Недавно я решила попытаться уменьшить свой. . .
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Как объединить две одинаковые БД Access с разными данными
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru