곧바로 예제부터 보자.
- #include <iostream>
- using namespace std;
- class AAA
- {
- char* str1;
- public:
- AAA(char* _str1){
- str1= new char[strlen(_str1)+1];
- strcpy(str1, _str1);
- }
- ~AAA(){ // virtual ~AAA()
- cout<<"~AAA() call!"<<endl;
- delete []str1;
- }
- virtual void ShowString(){
- cout<<str1<<' ';
- }
- };
- class BBB : public AAA
- {
- char* str2;
- public:
- BBB(char* _str1, char* _str2) : AAA(_str1){
- str2= new char[strlen(_str2)+1];
- strcpy(str2, _str2);
- }
- ~BBB(){
- cout<<"~BBB() call!"<<endl;
- delete []str2;
- }
- virtual void ShowString(){
- AAA::ShowString();
- cout<<str2<<endl;
- }
- };
- int main()
- {
- AAA * a=new BBB("Good", "evening");
- BBB * b=new BBB("Good", "morning");
- a->ShowString();
- b->ShowString();
- cout<<"-----객체 소멸 직전----"<<endl;
- delete a;
- delete b;
- return 0;
- }
위 코드를 보면, AAA클래스에서 생성자에서 동적 할당 하기에 소멸자에서 메모리 해제 하고 있고 마찬가지로 BBB클래스의 생성자에서
동적 할당을 하고 있어서 소멸자에서 메모리 해제 하고 있는 형태를 가지고 있다. BBB클래스의 객체가 소멸될때, AAA클래스의 소멸자도 호출이
된다.
BBB클래스 객체가 생성이 되면, AAA클래스의 생성자에 의해서도 메모리 공간 동적 할당 할것이고, BBB클래스도 마찬가지이다. 이
두곳에서 할당된 메모리 공간이 적절히 해제 될것이기에 우리가 신경 쓰지 않아도 되지만, 여전히 문제가 존재 한다.
문제제기 |
- int main()
- {
- //AAA * a=new BBB("Good", "evening");
- BBB * b=new BBB("Good", "morning");
- a->ShowString();
- b->ShowString();
- cout<<"-----객체 소멸 직전----"<<endl;
- //delete a;
- delete b;
- return 0;
- }
그럼 다음 코드를 보자.
- int main()
- {
- AAA * b=new BBB("Good", "morning");
- b->ShowString();
- cout<<"-----객체 소멸 직전----"<<endl;
- delete b;
- return 0;
- }
위의 경우를 생각해보자. BBB클래스는 AAA클래스를 상속하기 때문에 선언부가 3번째 줄처럼 바뀔수도 있을 것이다. 하지만
실행해보면, AAA클래스의 소멸자만 호출 되고 있음을 알 수 있다. 바로 이 부분에서 메모리의 유출이 발생된다.
AAA클래스의 포인터로 참조 하지만, 생성되는 객체는 B클래스의 객체이기 때문에, AAA,BBB 클래스에서도 생성자부분에서 메모리
공간이 동적 할당되는데는 문제가 없다. 하지만 객체가 소멸될때 A클래스 내에서 동적 할당한 메모리 공간은 반환되지만, BBB는 반환되지 않았던
것이다. 객체가 소멸하고자 했을때 소멸의 주체가 되는 포인터가 AAA클래스의 포인터였기 때문에 이런 일이 발생한다.
문제 해결 - 가상 소멸자 사용 |
- class AAA
- {
- public:
- virtual ~AAA(){ // virtual만 붙여주면 된다.
- cout<<"~AAA() call!"<<endl;
- delete []str1;
- };
virtual 소멸자의 경우 AAA클래스를 상속하고 있는 BBB클래스의 소멸자를 호출하게 된다. 이어서 BBB클래스가 AAA클래스를
상속하고 있으므로 AAA클래스의 소멸자를 호출하게 됨으로써 소멸자들이 정상적으로 호출되는 결과를 볼수 있다.
'개발지식창고 > C++' 카테고리의 다른 글
상속의 마지막, 다중 상속(Multiple Inheritance) (0) | 2010.11.14 |
---|---|
가상 함수 동작 원리와 단점 (0) | 2010.11.14 |
오버라이딩의 특징과 정적/동적 바인딩 (Overriding & Static / Dynamic Binding) (0) | 2010.11.14 |
오버라이딩 (Overriding (0) | 2010.11.14 |
상속 다섯번째, 상속된 객체와 참조와의 관계 (0) | 2010.11.14 |