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

Вызов деструктора. MSVS и MinGW - C++

Восстановить пароль Регистрация
 
greeezz
272 / 165 / 4
Регистрация: 10.07.2011
Сообщений: 441
09.01.2012, 00:06     Вызов деструктора. MSVS и MinGW #1
Всем доброго дня.

Вопрос в следующем.
К примеру имеем код:
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
//#include "stdafx.h"   //uncomment in ms visual studio
 
#include <iostream>
 
class Test{
public:
    Test(const int &);
    ~Test();
 
    Test operator*(const int&);
 
    int size;
    int *data;
};
 
Test::Test(const int &value) : size(value){
    data = new int[size];
    for(int i = 0; i < size; ++i){
        this->data[i] = i;
    }
}
 
Test Test::operator*(const int &right){
    Test temp(this->size);
    for(int i = 0; i < size; ++i){
        temp.data[i] = this->data[i] * right; 
    }
    return temp;
}
 
Test::~Test(){
        std::cout << "\n Delete object :: "<<this << "\n";
}
 
int main() {
 
    Test t(2);
    Test s(2);
 
    t*2;
 
    std::cin.get();
 
    return 0;
}
Пример работы программы :
После компиляции MSVS

Delete object :: 003AF5D0
Delete object :: 003AF63C
Delete object :: 003AF718
Delete object :: 003AF728


После компиляции MinGW

Delete object :: 0x28ff04
Delete object :: 0x28feec
Delete object :: 0x28fef4


Почему в первом случае диструстор вызывается 4 раза?

з.ы. я тут решал задачу по созданию класса матрицы.. так вот из-за такой разницы код по ссылке отлично работает после компиляции с MinGW но валится в рантайме после компиляции MSVS.

Всем заранее спаибо за участие. Что я делаю неправильно ?

тест на LWS
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
09.01.2012, 00:06     Вызов деструктора. MSVS и MinGW
Посмотрите здесь:

Вызов деструктора C++
C++ Явный вызов деструктора
C++ Не понятный вызов деструктора
C++ Вызов деструктора
C++ Повторный вызов деструктора
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
ForEveR
Модератор
Эксперт C++
 Аватар для ForEveR
7927 / 4709 / 318
Регистрация: 24.06.2010
Сообщений: 10,524
Завершенные тесты: 3
09.01.2012, 00:30     Вызов деструктора. MSVS и MinGW #2
greeezz,
C++
1
2
3
4
5
6
7
Test Test::operator*(const int &right){
        Test temp(this->size);
        for(int i = 0; i < size; ++i){
                temp.data[i] = this->data[i] * right; 
        }
        return temp;
}
При возврате вызывается конструктор копирования, затем деструктор для созданного в функции объекта. Возвращается скопированный объект. Т.к. используется указатель, но нет конструктора копирования - будут проблемы.
greeezz
272 / 165 / 4
Регистрация: 10.07.2011
Сообщений: 441
09.01.2012, 02:42  [ТС]     Вызов деструктора. MSVS и MinGW #3
ForEveR, я вроде понял. сейчас попробую дописать и сравнить результаты.

Добавлено через 1 час 51 минуту
добавил конструктор копирования.
C++
1
2
3
4
5
6
7
8
9
10
11
...
Test(const Test &);
...
 
Test::Test(const int &value) : size(value){
    this->data = new int[size];
    for(int i = 0; i < size; ++i){
        this->data[i] = i;
    }
}
....
Количество вызовов дестурктора в обоих случаях не изменилось. (я полагаю и не должно было).

Переформулирую вопрос.
Если я использую диструктор такого вида:
C++
1
2
3
Test::~Test(){
    delete [] this->data;
}
и хочу после выполнения умножения "присвоить" результат другому объекту типа Test
то как я понимаю наличие перегруженного оператора присвоения обязательно, потому что например в таком случае
C++
1
s = t*2;
сначала вызовется перегруженный оператор умножения который вернет объект типа Test содержащий результат вычислений. после чего будет вызван деструктор который освободит память созданного в операторе умножения объекта и только потом объекту s будет предпринята попытка присвоить адрес объекта возвращаемого перегруженным оператором умножения, а этот объект уже фактически не существует. Результат в s будет мусор.
Теперь я перегрузил оператор присвоения.
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
    
.... 
Test &operator=(const Test &);
.....
Test &Test::operator=(const Test &right){
    if(this != &right){
        if(this->size!=right.size){
            delete [] this->data;
            this->size = right.size;
            this->data = new int[right.size];
        }
        for(int i = 0; i < size; ++i){
            this->data[i] = right.data[i]; 
        }
    }
    return *this;
}
Мой ход мыслей верен ?
Я запускал дебаг в eclipse. прошел пошагово по всему коду и у меня ни разу не был вызван конструктор копирования.... непоняяяятно
(простите за много букв )

КОД В ИТОГЕ

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
#include <iostream>
 
class Test{
public:
    Test(const int &);
    
    Test(const Test &);
    
    ~Test();
 
    Test operator*(const int &);
    Test &operator=(const Test &);
 
    int size;
    int *data;
};
 
Test::Test(const int &value) : size(value){
    this->data = new int[size];
    for(int i = 0; i < size; ++i){
        this->data[i] = i;
    }
}
 
Test::Test(const Test &right) : size(right.size){
    this->data = new int[right.size];
    for(int i = 0; i < this->size; ++i){
        this->data[i] = right.data[i]; 
    }
}
 
Test Test::operator*(const int &right){
    Test temp(this->size);
    for(int i = 0; i < size; ++i){
        temp.data[i] = this->data[i] * right; 
    }
    return temp;
}
 
Test &Test::operator=(const Test &right){
    if(this != &right){
        if(this->size!=right.size){
            delete [] this->data;
            this->size = right.size;
            this->data = new int[right.size];
        }
        for(int i = 0; i < size; ++i){
            this->data[i] = right.data[i]; 
        }
    }
    return *this;
}
 
Test::~Test(){
    delete [] this->data;
}
 
int main() {
 
    Test t(2);
    Test s(5);
 
    s = t*2;
 
 
    for(int i = 0; i < s.size; ++i){
    std::cout << s.data[i] << " ";
    }
 
    std::cin.get();
 
    return 0;
}
ForEveR
Модератор
Эксперт C++
 Аватар для ForEveR
7927 / 4709 / 318
Регистрация: 24.06.2010
Сообщений: 10,524
Завершенные тесты: 3
09.01.2012, 04:03     Вызов деструктора. MSVS и MinGW #4
greeezz, М...

Есть некий оператор умножения, возвращающий объект некого класса.
В перегрузке данного оператора создается объект класса - встречается return, вызывается конструктор копирования, скопированный объект возвращается, созданный объект удаляется, вызывается оператор присваивания, чтобы присвоить возвращенный объект.
ForEveR
Модератор
Эксперт C++
 Аватар для ForEveR
7927 / 4709 / 318
Регистрация: 24.06.2010
Сообщений: 10,524
Завершенные тесты: 3
09.01.2012, 04:18     Вызов деструктора. MSVS и MinGW #5
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
#include <iostream>
#include <cstdlib>
 
class Cl
{
public:
    Cl(int value = 0):val(value)
    {
        std::cout << "Construct object with value: " << val << std::endl;
    }
    ~Cl()
    {
        std::cout << "Destruct object with value: " <<  val << std::endl;
    }
    Cl(const Cl& rhs):val(rhs.val)
    {
        std::cout << "Construct object from other object" << std::endl;
    }
    Cl& operator =(const Cl& rhs)
    {
        std::cout << "Assignment operator" << std::endl;
        Cl tmp(rhs);
        swap(tmp);
        return *this;
    }
    Cl operator *(const int value)
    {
        Cl tmp(*this);
        tmp.val *= value;
        return tmp;
    }
private:
    int val;
    void swap(Cl& rhs)
    {
        std::swap(val, rhs.val);
    }
};
 
int main()
{
    Cl val(1);
    Cl tmp;
    tmp = val * 2;
    system("pause");
}
Без оптимизаций. Вывод.
Миниатюры
Вызов деструктора. MSVS и MinGW  
greeezz
272 / 165 / 4
Регистрация: 10.07.2011
Сообщений: 441
09.01.2012, 07:37  [ТС]     Вызов деструктора. MSVS и MinGW #6

Не по теме:

колдую..........



Добавлено через 2 часа 36 минут
ForEveR, Спасибо, отличный пример! Я разобрался наколдавал!
Yandex
Объявления
09.01.2012, 07:37     Вызов деструктора. MSVS и MinGW
Ответ Создать тему
Опции темы

Текущее время: 16:45. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru