36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
1

При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О

28.10.2013, 14:18. Показов 5169. Ответов 27
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Вот такой код
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class A
{
public:
    A(){}
    virtual ~A(){}
};
 
class C
{
public:
    C(A a){}
    ~C(){}
    C(C& c){}
};
 
int main()
{
    C c=C(A());
 
    system("pause");
    return 0;
}
Если посмотреть в отладчике, то видно, что конструктор C() вызывается 2 раза, причём второй раз конструктор копирования. Я понимаю, если бы это было с конструктором A(), т. к. A() передаётся по значению. Но ни могу понять почему именно конструктор C(). Причём если деструктор класса A сделать не виртуальным, то конструктор класса C() будет вызываться 1 раз. С чем это связано?
Миниатюры
При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О  
0
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
28.10.2013, 14:18
Ответы с готовыми решениями:

Конструктор вызывается один раз, а деструктор два раза
Программа для обработки исключений. Вообщем если ввести два числа и второе будет 0, то конструктор...

Почему при помещении объекта в list вызывается конструктор и деструктор?
Нужна помощь опытного программиста. Не могу понять почему при помещение объекта в list вызывается...

Два раза вызывается деструктор
Создал класс, в процессе выполнения программы вызвал деструктор, но когда программа завершается,...

Виртуальный деструктор вызывается два раза
Здравствуйте, у меня есть базовый и производный классы с динамическими массивами, я создаю массив...

27
5498 / 4893 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
28.10.2013, 14:41 2
Цитата Сообщение от Виктор_Сен Посмотреть сообщение
Но ни могу понять почему именно конструктор C().
При инициализации объекта (с) другим объектом (созданным с помощью C(A())), вызывается конструктор копирования.
0
46 / 46 / 18
Регистрация: 25.10.2011
Сообщений: 183
28.10.2013, 14:43 3
1. Вызывается конструктор А
2. Вызывается конструктор копий С
3. Вызывается деструктор А
Что странного?
0
5498 / 4893 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
28.10.2013, 14:51 4
С виртуальностью - 1 скрин, и без виртуальности - 2 скрин.
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
#include <iostream>
using namespace std;
 
class A
{
public:
    A(){cout << "A()" << endl;}
    virtual ~A(){cout << "~A()" << endl;}
};
 
class C
{
public:
    C(A a){cout << "C(A a)" << endl;}
    ~C(){ cout << "~C()" << endl;}
    C(C& c){ cout << "C(C &c)" << endl;}
};
 
int  main()
{
    C c = C(A());
 
    system("pause");
    return 0;
}
Миниатюры
При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О   При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О  
0
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 14:51  [ТС] 5
Цитата Сообщение от Hunter13ua Посмотреть сообщение
1. Вызывается конструктор А
2. Вызывается конструктор копий С
3. Вызывается деструктор А
Что странного?
То, что Вы написали, НЕ странно. Странно то, что вызывается 2 раза конструктор для объекта C. На левом скриншоте это выделено
Assembler
1
2
3
4
5
6
call C::C
...
call C::C
...
call C::~C
...
0
5498 / 4893 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
28.10.2013, 14:55 6
Цитата Сообщение от Виктор_Сен Посмотреть сообщение
Странно то, что вызывается 2 раза конструктор для объекта C.
Что в этом странного? Первый раз просто конструктор: C(A()), второй раз конструктор копирования, для инициализации этим, созданным объектом, объекта c.
Миниатюры
При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О  
0
шКодер самоучка
2227 / 1921 / 927
Регистрация: 09.10.2013
Сообщений: 4,256
Записей в блоге: 7
28.10.2013, 15:02 7
C++
1
2
3
4
5
6
// код
C c = C(A());
// равносилен
A a;
C temp(a);
C c(temp);
0
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 15:14  [ТС] 8
Цитата Сообщение от alsav22 Посмотреть сообщение
С виртуальностью - 1 скрин, и без виртуальности - 2 скрин.
Цитата Сообщение от alsav22 Посмотреть сообщение
При инициализации объекта (с) другим объектом (созданным с помощью C(A())), вызывается конструктор копирования.
Тут другого объекта нет. Тут только объект (с). Получается, что создаётся временный объект C(A())), потом от него инициализируется уже (c). Вот я и не понимаю, зачем это нужно, если при первом вызове конструктора объект уже есть... Как я полагаю, конструктор копирования срабатывает при инициализации объекта уже готовым объектом. Но тут получается заранее готового объекта нет. Он создаётся только для того, чтобы вызвать конструктор копирования, зачем это нужно, если при первом вызове конструктора у нас фактически уже есть нужный объект. И потом без виртуальности этого не происходит, т. е. конструктор для (C) вызывается 1 раз, значит в принципе конструктор копирования не нужен здесь, это ведь не не инициализация от готового объекта по сути, это просто инициализация через вызов конструктора. Что такого есть в виртуальности, что компилятор генерирует вызов ещё и конструктора копирования?

Добавлено через 5 минут
Цитата Сообщение от alsav22 Посмотреть сообщение
Что в этом странного? Первый раз просто конструктор: C(A()), второй раз конструктор копирования, для инициализации этим, созданным объектом, объекта c
Зачем это нужно, если можно вызвать конструктор (не копирования) сразу для (С)?

Добавлено через 1 минуту
Цитата Сообщение от Cra3y Посмотреть сообщение
C++
1
2
3
4
5
6
// код
C c = C(A());
// равносилен
A a;
C temp(a);
C c(temp);
Если бы было так, то без виртуальности вывод на левом скриншоте и правом совпадали бы. Тут ключевой момент именно в виртуальности. Без неё конструктор копирования не вызывается.

Добавлено через 2 минуты
И если передавать класс (A) по ссылке, то и с виртуальностью, и без виртуальности конструктор копирования не вызывается. Значит всё-таки код
C++
1
C c = C(A());
и
C++
1
2
3
A a;
C temp(a);
C c(temp);
не равносильны.
0
46 / 46 / 18
Регистрация: 25.10.2011
Сообщений: 183
28.10.2013, 15:14 9
Так зачем писать было
C++
1
C c = C(A());
Что бы работало, как я написал:
C++
1
C c(A());
Или я чего-то не понимаю в вашем вопросе?
0
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 15:20  [ТС] 10
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
#include <iostream>
using namespace std;
 
class A
{
public:
    A(){cout<<"A()"<<endl;}
    virtual ~A(){cout<<"~A()"<<endl;}
};
 
class C
{
public:
    C(A& a){cout<<"C(A())"<<endl;}
    ~C(){cout<<"~C()"<<endl;}
    C(C& c){cout<<"C(C& c)"<<endl;}
};
 
int main()
{
    C c=C(A());
 
    system("pause");
    return 0;
}
Такой код например уже даёт вывод
A()
C(A())
~A()

Добавлено через 2 минуты
Цитата Сообщение от Hunter13ua Посмотреть сообщение
Что бы работало, как я написал:
Код C++
1
C c(A());
Так вообще ни какого вывода нет, т. к. вот что:
Предупреждение 1 warning C4930: C c(A (__cdecl *)(void)): функция с прототипом не вызвана (предполагалось определение переменной?) D:\documents\Visual Studio 2010\Projects\LSD\LSD\main.cpp 26 1 LSD
0
205 / 165 / 41
Регистрация: 25.10.2013
Сообщений: 527
28.10.2013, 15:22 11
Вам для полного понимания, надо было бы ещё в A указать ещё конструктор копирования, а то используется конструктор по умолчанию, который ничего не выводит.
0
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 15:25  [ТС] 12
Цитата Сообщение от Hunter13ua Посмотреть сообщение
Или я чего-то не понимаю в вашем вопросе?
Всё-таки при определении переменной нужен знак равенства, и при определении переменной, как я понимаю, этот знак равенства эквивалентен не копированию, а созданию "с нуля" переменной, если справа от знака равенства нет уже существующей переменной (которая связана с буквой, её обозначающей).
0
шКодер самоучка
2227 / 1921 / 927
Регистрация: 09.10.2013
Сообщений: 4,256
Записей в блоге: 7
28.10.2013, 15:28 13
У меня это не компилится %)
Код
max@notebook:~/src/tests$ g++ c.cpp
c.cpp: In function ‘int main()’:
c.cpp:21:16: error: no matching function for call to ‘C::C(A)’
     C c = C(A());
                ^
c.cpp:21:16: note: candidates are:
c.cpp:16:5: note: C::C(C&)
     C(C& c){ cout << "C(C &c)" << endl;}
     ^
c.cpp:16:5: note:   no known conversion for argument 1 from ‘A’ to ‘C&’
c.cpp:14:5: note: C::C(A&)
     C(A &a){cout << "C(A a)" << endl;}
     ^
c.cpp:14:5: note:   no known conversion for argument 1 from ‘A’ to ‘A&’
max@notebook:~/src/tests$ g++ --version
g++ (Ubuntu/Linaro 4.8.1-10ubuntu8) 4.8.1
0
В астрале
Эксперт С++
8049 / 4806 / 655
Регистрация: 24.06.2010
Сообщений: 10,562
28.10.2013, 15:35 14
C++
1
C c=C(A());
Есть инициализировать объект C объектом, получившемся из использования C(A()), используя конструктор копирования (или не используя, зависит от copy-elision).
Без конструктора копирования нужно:
C++
1
C c((A()));
Ну и да, код не скомпилируется, т.к. на временную переменную нельзя создать ссылку (скомпилируется только в студии).
0
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 15:35  [ТС] 15
У меня нормально компилится
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
#include <iostream>
using namespace std;
 
class A
{
public:
    A(){cout<<"A()"<<endl;}
    virtual ~A(){cout<<"~A()"<<endl;}
    A(A& a){cout<<"A(A& a)"<<endl;}
};
 
class C
{
public:
    C(A a){cout<<"C(A())"<<endl;}
    ~C(){cout<<"~C()"<<endl;}
    C(C& c){cout<<"C(C& c)"<<endl;}
};
 
int main()
{
    C c=C(A());
 
    system("pause");
    return 0;
}
Вот вывод
Миниатюры
При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О  
0
5498 / 4893 / 831
Регистрация: 04.06.2011
Сообщений: 13,587
28.10.2013, 15:36 16
Цитата Сообщение от Виктор_Сен Посмотреть сообщение
Получается, что создаётся временный объект C(A())), потом от него инициализируется уже (c). Вот я и не понимаю, зачем это нужно, если при первом вызове конструктора объект уже есть...
Потому что код так написан. Спросите у того, кто его писал, зачем он так сделал.
0
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 15:38  [ТС] 17
Цитата Сообщение от ForEveR Посмотреть сообщение
Без конструктора копирования нужно:
Код C++
1
C c((A()));
Так у меня вообще ни какой конструктор не вызывается. И варинг появляется:
Предупреждение 1 warning C4930: C c(A (__cdecl *)(void)): функция с прототипом не вызвана (предполагалось определение переменной?) D:\documents\Visual Studio 2010\Projects\LSD\LSD\main.cpp 27 1 LSD

Добавлено через 1 минуту
Цитата Сообщение от alsav22 Посмотреть сообщение
Потому что код так написан. Спросите у того, кто его писал, зачем он так сделал.
А по-другом вообще конструктор не вызывается.
0
В астрале
Эксперт С++
8049 / 4806 / 655
Регистрация: 24.06.2010
Сообщений: 10,562
28.10.2013, 15:39 18
Виктор_Сен, На скобки смотрите внимательнее. Точнее на кол-во скобок.
1
36 / 29 / 2
Регистрация: 01.08.2011
Сообщений: 176
28.10.2013, 15:42  [ТС] 19
Но без виртуальности вызова конструктора копирования нет. В этом видимо причина.
0
В астрале
Эксперт С++
8049 / 4806 / 655
Регистрация: 24.06.2010
Сообщений: 10,562
28.10.2013, 15:42 20
А вообще вот, почитайте: Как работает компилятор при создании объекта недавно была тема именно про это.
1
28.10.2013, 15:42
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
28.10.2013, 15:42
Помогаю со студенческими работами здесь

Почему деструктор вызывается два раза
Всем привет. Есть код, для примера #include &lt;vector&gt; using namespace std; struct AA { int n;...

Конструктор вызывается 2 раза
Помогите пожалуйста найти ошибку. Конструктор вызывается 2 раза. #include &lt;iostream&gt; #include...

Когда вызывается деструктор класса?
Вопрос собственно в том, когда происходит освобождение памяти, занимаемой объектом класса? Допустим...

При динамическом создании объекта в классе деструктор не вызывается
Добрый день еще раз. Помогите заполнить пробелы в ООП) Есть три класса: // Вспомогательный класс...


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

Или воспользуйтесь поиском по форуму:
20
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2023, CyberForum.ru