=================================
=================================
=================================
출처: http://boxbop.tistory.com/64
아주 오~~~랜만에 포스팅합니다! 너무 오래되서 전에 무슨 강을 공부했는지도 기억이 가물해서...;; 역시 사람의 망각 곡선은 무시할게 못되죠...ㅠㅠㅠㅠ 이럴때는 천재들이 부럽네요! 무튼 오늘은 virtual 의 원리와 다중상속에 대해서 공부해보도록 하겠습니다.
[가상 함수의 기본 원리]
가상 함수가 동작하는 원리에 대해서 간단하게 알아보겠습니다.
#include <iostream>
using std::endl;
using std::cout;
class A
{
int a;
int b;
public:
virtual void fct1() { cout<<"fct1(...)"<<endl; }
virtual void fct2() { cout<<"fct2(...)"<<endl; }
};
class B : public A
{
int c;
int b;
public:
virtual void fct1() { cout<<" overriding fct1(...)"<<endl; }
void fct3() { cout<<"fct3(...)"<<endl; }
};
int main(void)
{
A* aaa = new A();
aaa->fct1();
B* bbb = new B();
bbb->fct1();
return 0;
}
먼저 A 클래스를 보면 virtual로 선언된 2개의 가상함수가 존재합니다. 마찬가지로 B 클래스에도 하나의 가상함수가 존재하지요. 중요한 부분은 B 클래스가 A 클래스를 상속했습니다만 fct1() 이라는 함수명이 중복되어 오버라이딩이 됩니다. 당연히 B 클래스 입장에서 fct1() 함수를 호출하면 "overriding fct1(...)"이 출력이 되겠죠? 마찬가지로 B 클래스 입장에서 fct2() 함수를 오출하면 A 클래스를 상속했으니 A 클래스의 fct2() 함수를 호출하게 되겠구요~
즉, B 클래스의 입장에서 보게되면 오버라이딩된 A 클래스의 가상 함수 fct1에 대한 정보가 없기 때문에 B 클래스의 fct1 함수가 대신 호출 되는 것입니다. 이것이 바로 가상 함수의 기본원리 입니다.
[다중 상속에 대한 이해]
결론 부터 말씀드리자면 다중 상속은 왠만하면 피해하는게 좋습니다. 이유를 간단하게 얘기하자면 클래스들의 관계가 복잡해지고, 관리하기에도 어렵습니다.
#include <iostream>
using std::cout;
using std::endl;
class AAA{
public:
void String1() { cout<< " AAA::String1"<<endl; }
};
class BBB{
public:
void String2() { cout<<" BBB::String2"<<endl; }
};
class CCC : public AAA, public BBB{
public:
void ShowString()
{
String1();
String2();
}
};
int main(void)
{
CCC ccc;
ccc.ShowString();
return 0;
}
CCC 클래스는 AAA 클래스와 BBB 클래스를 동시에 상속하고 있습니다. 결과 값은 예상할 수 있겠죠? 딱히 문제되는 부분 없습니다. 자 여기서 다중 상속을 할 경우 주의해야 할 사항이 몇 가지 있습니다. 다음 예제를 통해서 살펴보도록 하겠습니다.
#include <iostream>
using std::cout;
using std::endl;
class AAA{
public:
void String() { cout<< " AAA::String"<<endl; }
};
class BBB{
public:
void String() { cout<<" BBB::String"<<endl; }
};
class CCC : public AAA, public BBB{
public:
void ShowString()
{
String();
String();
}
};
int main(void)
{
CCC ccc;
ccc.ShowString();
return 0;
}
AAA 클래스와 BBB 클래스가 지니고 있는 맴버 함수의 이름이 똑같습니다. 때문에 컴파일 에러가 발생하게 됩니다. 무엇을 호출해야될지 모르기 때문이죠. 때문에 다음과 같이 변경해주어야 합니다.
String() ----> AAA::String();
String() ----> BBB::String();
위와 같은 다중상속의 모호성은 이렇게 쉽게 해결해보았습니다.
#include <iostream>
using std::cout;
using std::endl;
class AAA{
public:
void String1() { cout<< " AAA::String"<<endl; }
};
class BBB : public AAA{
public:
void String2() { cout<<" BBB::String"<<endl; }
};
class CCC : public AAA{
public:
void String3() { cout<<" CCC::String"<<endl; }
};
class DDD : public BBB, public CCC{
public:
void ShowString()
{
String1();
String2();
String3();
}
};
int main(void)
{
DDD ddd;
ddd.ShowString();
return 0;
}
DDD 클래스가 결과적으로 AAA 클래스를 두번이나 상속하게 되어있습니다. 이게 바로 문제가 되는 부분입니다. 즉 AAA 클래스의 맴버 함수 String1 을 BBB 클래스를 통해서, 그리고 CCC 클래스를 통해서 상속받고 있게되죠. DDD 클래스의 볼드 처리된 부분을 보시면 이 String1이 BBB 클래스를 통해서 상속받은 함수인지 CCC 클래스를 통해서 상속받은 함수인지 어떠한 함수를 호출할지 몰라서 결국 컴파일 에러를 발생시킵니다.
이러한 문제는 virtual 상속을 통해서 간단하게 해결 할 수 있습니다.
class BBB : virtual public AAA{
public:
void String2() { cout<<" BBB::String"<<endl; }
};
class CCC : virtual public AAA{
public:
void String3() { cout<<" CCC::String"<<endl; }
};
둘다 모두 AAA 클래스를 virtual 상속하고 있습니다. 반드시 둘다 virtual 상속을 해야합니다. 이제부터는 BBB 클래스와 CCC 클래스를 다중 상속한다 하더라도, AAA 클래스 안에 존재하는 맴버들은 한번만 상속이 이뤄지게 됩니다. 이것이 바로 virtaul 상속을 하는 이유입니다.
결론은 다중 상속은 클래스의 관계를 복잡하게 흐려놓습니다. 다중 상속으로 밖에 해결이 안 되는 문제도 존재하지 않는다고 하니까 가급적이면 사용하지 않는 편이 좋다고하죠^^;; 오랜만에 포스팅하니까 현기증이..... 이번강은 여기서 마치도록 하겠습니다.
출처: http://boxbop.tistory.com/64 [boxbop]
=================================
=================================
=================================
출처: http://itguru.tistory.com/211
안녕하세요 여러분. 지난 강좌에서는 놀라움의 연속이었던 virtual 키워드의 기능에 대해서 설명하였습니다. virtual 키워드를 통해서 동적 바인딩이라는 것을 이루어 낼 수 있었지요. 이번 강좌에서는 가상 함수와 상속에 관련하여 잡다한 내용들을 모두 짚고 넘어가도록 하겠습니다.
지난 시간에 배웠던 것을 간단히 정리해보자면 다음과 같습니다. Parent 클래스와 Child 클래스에 모두 f 라는 가상함수가 정의되어 있고, Child 클래스가 Parent 를 상속 받는다고 해봅시다. 그런 다음에 동일한 Parent* 타입의 포인터들도 각각 Parent 객체와 Child 객체를 가리킨다고 해봅시다.
Parent* p = new Parent();
Parent* c = new Child();
컴퓨터 입장에서 p 와 c 모두 Parent 를 가리키는 포인터들이므로, 당연히
p->f();
c->f();
를 했을 때 모두 Parent 의 f() 가 호출되어야 하겠지만, 실제로는 f 가 가상함수므로, '실제로 p 와 c 가 가리키는 객체의' f, 즉 p->f() 는 Parent 의 f 를, c->f() 는 Child 의 f 가 호출됩니다. 이와 같은 일이 가능한 이유는 f 를 가상함수로 만들었기 때문입니다.
virtual 소멸자 |
사실 클래스의 상속을 사용함으로써 중요하게 처리해야 되는 부분이 있습니다. 바로, 소멸자를 가상함수로 만들어야 된다는 점입니다.
#include <iostream>
using namespace std;
class Parent
{
public :
Parent()
{
cout << "Parent 생성자 호출" << endl;
}
~Parent()
{
cout << "Parent 소멸자 호출" << endl;
}
};
class Child : public Parent
{
public:
Child() : Parent()
{
cout << "Child 생성자 호출" << endl;
}
~Child()
{
cout << "Child 소멸자 호출" << endl;
}
};
int main()
{
cout << "--- 평범한 Child 만들었을 때 ---" << endl;
{
Child c;
}
cout << "--- Parent 포인터로 Child 가리켰을 때 ---" << endl;
{
Parent *p = new Child();
delete p;
}
}
성공적으로 컴파일 하였다면
와 같이 나옵니다.
일단 평범하게 Child 객체를 만든 부분을 살펴봅시다.
cout << "--- 평범한 Child 만들었을 때 ---" << endl;
{
Child c;
}
생성자와 소멸자의 호출 순서를 살펴보자면, Parent 생성자 → Child 생성자 → Child 소멸자 → Parent 소멸자 순으로 호출됨을 알 수 있습니다. 이와 같은 과정이 당연한 이유는 객체를 만들고 소멸시키는 일을 집을 짓고 철거하는 일로 비유할 수 있습니다. 집을 지을 때에는 큰 틀, 즉 기초공사를 하고 건물을 세운 다음에 (Parent 생성자 호출), 집 내부 공사 - 인테리어, 가구 배치 등을 하게 됩니다 (Child 생성자 호출). 그리고 역으로 집을 철거할 때에는 안에 있는 내용물들을 모두 제거한 뒤에 (Child 소멸자 호출), 집 구조물을 철거하겠지요 (Parent 소멸자 호출).
그런데 문제는 그 아래 Parent 포인터가 Child 객체를 가리킬 때 입니다.
cout << "--- Parent 포인터로 Child 가리켰을 때 ---" << endl;
{
Parent *p = new Child();
delete p;
}
delete p 를 하더라도, p 가 가리키는 것은 Parent 객체가 아닌 Child 객체 이기 때문에, 위에서 보통의 Child 객체가 소멸되는 것과 같은 순서로 생성자와 소멸자들이 호출되어야만 합니다. 그런데 실제로는, Child 소멸자가 호출되지 않습니다.
소멸자가 호출되지 않는다면 여러가지 문제가 생길 수 있습니다. 예를 들어서, Child 객체에서 메모리를 동적으로 할당하고 소멸자에서 해제하는데, 소멸자가 호출 안됬다면 메모리 누수(memory leak)가 생기겠지요. 하지만 virtual 키워드를 배운 이상 여러분은 무엇을 해야 하는지 알고 계실 것입니다. 단순히 Parent 의 소멸자를 virtual 로 만들어버리면 됩니다. Parent 의 소멸자를 virtual 로 만들면, p 가 소멸자를 호출할 때, Child 의 소멸자를 성공적으로 호출할 수 있게 됩니다.
#include <iostream>
using namespace std;
class Parent
{
public :
Parent()
{
cout << "Parent 생성자 호출" << endl;
}
virtual ~Parent()
{
cout << "Parent 소멸자 호출" << endl;
}
};
class Child : public Parent
{
public:
Child() : Parent()
{
cout << "Child 생성자 호출" << endl;
}
~Child()
{
cout << "Child 소멸자 호출" << endl;
}
};
int main()
{
cout << "--- 평범한 Child 만들었을 때 ---" << endl;
{
Child c;
}
cout << "--- Parent 포인터로 Child 가리켰을 때 ---" << endl;
{
Parent *p = new Child();
delete p;
}
}
와 같이 제대로 Child 소멸자가 호출됨을 알 수 있습니다.
여기서 한 가지 질문을 하자면, 그렇다면 왜 Parent 소멸자는 호출이 되었는가 인데, 이는 Child 소멸자를 호출하면서, Child 소멸자가 '알아서' Parent 의 소멸자도 호출해주기 때문입니다 (Child 는 자신이 Parent 를 상속받는다는 것을 알고 있습니다). 반면에 Parent 소멸자를 먼저 호출하게 되면, Parent 는 Child 가 있는지 없는지 모르므로, Child 소멸자를 호출해줄 수 없습니다 (Parent 는 자신이 누구에서 상속해주는지 알 수 없지요).
이와 같은 연유로, 상속될 여지가 있는 Base 클래스들은 (위 경우 Parent), 반드시 소멸자를 virtual 로 만들어주어야 나중에 문제가 발생할 여지가 없게 됩니다.
레퍼런스도 된다 |
여태 까지 부모 클래스에서 자식 클래스의 함수에 접근할 때 항상 부모 클래스의 포인터를 통해서 접근하였습니다. 하지만, 사실 부모 클래스의 레퍼런스여도 문제 없이 작동합니다. 아래 간단한 예제를 통해 살펴보겠습니다.
#include <iostream>
using namespace std;
class A {
public :
virtual void show() {
cout << "Parent !" << endl;
}
};
class B : public A {
public:
void show() {
cout << "Child!" << endl;
}
};
void test(A& a) {
a.show();
}
int main()
{
A a;
B b;
test(a);
test(b);
return 0;
}
성공적으로 컴파일 하였다면
와 같이 나옵니다.
void test(A& a) {
a.show();
}
test 함수를 살펴보면 A 클래스의 레퍼런스를 받게 되어 있지만,
test(b);
를 통해서 B 클래스의 객체를 전달하였는데도 잘 작동하였습니다. 이는, B 클래스가 A 클래스를 상속 받고 있기 때문입니다. 즉, 함수에 타입이 부모 클래스여도 그 자식 클래스는 타입 변환되어 전달 할 수 있습니다.
따라서 test 함수에서 show() 를 호출하였을 때 인자로 b 를 전달하였다면, 비록 전달된 인자가 A의 객체라고 표현되어 있지만 show 함수가 virtual 으로 정의되어 있기 때문에 알아서 B 의 show 함수를 찾아내서 호출하게 됩니다. 물론 test 에 a 를 전달하였을 때에는 A 의 show 함수가 호출되겠지요.
가상 함수의 구현 원리 |
여태 까지 virtual 키워드의 능력을 본 바로는 이러한 의문이 들 수 도 있을 것입니다.
그냥 그럼 모든 함수들을 virtual 로 만들어버리면 안되나?
사실 이는 매우 좋은 질문입니다. 왜냐하면 모든 함수들을 virtual 로 만들어버린다고 해서 문제될 것이 전혀 없기 때문입니다. 간혹 '가상' 이라는 이름 때문에 혼동하시는 분이 계시는데, virtual 키워드를 붙여서 가상 함수로 만들었다 해도 실제로 존재하는 함수이고 정상적으로 호출도 할 수 있습니다. 또한 모든 함수들을 디폴트로 가상 함수로 만듬으로써, 언제나 동적 바인딩이 제대로 동작하게 만들 수 있습니다.
실제로 자바의 경우 모든 함수들이 디폴트로 virtual 함수로 선언됩니다.
그렇다면 왜 C++ 에서는 virtual 키워드를 이용해 사용자가 직접 virtual 로 선언하도록 하였을까요? 그 이유는 가상 함수를 사용하게 되면 약간의 오버헤드 (overhead) 가 존재하기 때문입니다. 즉, 보통의 함수를 호출하는 것 보다 가상 함수를 호출하는 데 걸리는 시간이 (아주아주 조금) 더 오래 걸립니다. 이를 이해하기 위해 가상 함수라는 것이 어떻게 구현되는지, 다시 말해 마술과 같은 동적 바인딩이 어떻게 구현되는지 살펴보도록 합시다.
예를 들어서 다음과 같은 간단한 두 개의 클래스를 생각해봅시다.
class Parent
{
public:
virtual void func1();
virtual void func2();
};
class Child : public Parent
{
public:
virtual void func1();
void func3();
};
위와 같이 구성됩니다. 가상 함수와 가상 함수가 아닌 함수와의 차이점을 살펴보자면 Child 의 func3() 같이 비 가상함수들은 그냥 단순히 특별한 단계를 걸치지 않고, func3() 을 호출하면 직접 실행됩니다. 하지만, 가상 함수를 호출하였을 때는 그 실행 과정이 다릅니다. 위에서도 보이다 싶이, 가상 함수 테이블을 한 단계 더 걸쳐서, 실제로 '어떤 함수를 고를지' 결정하게 됩니다. 예를 들어서;
Parent* p = Parent();
p->func1();
을 해봅시다. 그러면, 컴파일러는
1. p 가 Parent 를 가리키는 포인터 이니까, func1() 의 정의를 Parent 클래스에서 찾아봐야겠다.
2. func1() 이 가상함수네? 그렇다면 func1() 을 직접 실행하는게 아니라, 가상 함수 테이블에서 func1() 에 해당하는 함수를 실행해야겠다.
그리고 실제로 프로그램 실행시에, 가상 함수 테이블에서 func1() 에 해당하는 함수(Parent::func1()) 을 호출하게 됩니다.
에 해당하는 코드를 작성하게 됩니다. 그렇다면, 다음의 경우는 어떨까요?
Parent* c = Child();
c->func1();
위 처럼 똑같이 프로그램 실행시에 가상 함수 테이블에서 func1() 에 해당하는 함수를 호출하게 되는데, 이번에는 p 가 실제로는 Child 객체를 가리키고 있으므로, Child 객체의 가상 함수 테이블을 참조하여, Child::func1() 을 호출하게 됩니다. 이는 성공적으로 Parent::func1() 을 오버라이드 하지요.
이와 같이 두 단계에 걸쳐서 함수를 호출함을 통해 소프트웨어적으로 동적 바인딩을 구현할 수 있게 됩니다. 이러한 이유로 가상 함수를 호출하는 경우, 일반적인 함수 보다 약간 더 시간이 오래 걸리게 됩니다. 물론 눈부시게 CPU 의 속도가 빨라짐에 따라서 이러한 차이는 극히 미미해졌으나, 최적화가 매우 중요한 분야에서는 이를 감안할 필요가 있습니다. 아무튼 이러한 연유로 인해, 다른 언어들과는 다르게, C++ 에서는 멤버 함수가 디폴트로 가상함수가 되도록 설정하지는 않습니다.
순수 가상 함수(pure virtual function)와 추상 클래스(abstract class) |
#include <iostream>using namespace std;class Animal{public:Animal() {}virtual ~Animal() {}virtual void speak() = 0;};class Dog : public Animal{public:Dog() : Animal() {}void speak() {cout << "왈왈" << endl;}};class Cat : public Animal{public:Cat() : Animal() {}void speak() {cout << "야옹야옹" << endl;}};int main(){Animal* dog = new Dog();Animal* cat = new Cat();dog->speak();cat->speak();}
위 코드를 보면서 한 가지 특이한 점을 눈치 채셨을 것입니다.
class Animal{public:Animal() {}virtual ~Animal() {}virtual void speak() = 0;};
Animal a;a.speak();
error C2259: 'Animal' : cannot instantiate abstract class1> due to following members:1> 'void Animal::speak(void)' : is abstract
class Dog : public Animal{public:Dog() : Animal() {}void speak() {cout << "왈왈" << endl;}};
예를 들어서 위에서 예를 든 Animal 클래스의 경우
class Animal{public:Animal() {}virtual ~Animal() {}virtual void speak() = 0;};
추상 클래스의 또 한가지 특징은 비록 객체는 생성할 수 없지만, 추상 클래스를 가리키는 포인터는 문제 없이 만들 수 있다는 점입니다. 위 예에서도 살펴보았듯이, 아무런 문제 없이 Animal* 의 변수를 생성하였습니다.
Animal* dog = new Dog();Animal* cat = new Cat();dog->speak();cat->speak();
다중 상속(multiple inheritance) |
마지막으로 C++ 에서의 상속의 또 다른 특징인 다중 상속에 대해 알아보도록 합시다. C++ 에서는 한 클래스가 다른 여러 개의 클래스들을 상속 받는 것을 허용합니다. 이를 가리켜서 다중 상속 (multiple inheritance) 라고 부릅니다.
class A
{
public:
int a;
};
class B
{
public:
int b;
};
class C : public A, public B
{
public:
int c;
};
위 경우, 클래스 C 가 A 와 B 로 부터 동시에 같이 상속 받고 있습니다.
이를 그림으로 표현하자면 위 같은 모양이 되겠지요. 사실 다중 상속은 보통의 상속 하고 똑같이 생각하시면 됩니다. 단순히 그냥 A 와 B 의 내용이 모두 C 에 들어간다고 생각하시면 됩니다. 따라서;
C c;
c.a = 3;
c.b = 2;
c.c = 4;
와 같은 것이 가능하게 되는 것이지요. 다중 상속에서 한 가지 재미있는 점은 생성자들의 호출 순서 입니다. 여러분은 과연 위 예에서 A 의 생성자가 먼저 호출될지, B 의 생성자가 먼저 호출될 지 궁금할 것입니다. 한 번 확인을 해보도록 하겠습니다.
#include <iostream>
using namespace std;
class A
{
public:
int a;
A() { cout << "A 생성자 호출" << endl;}
};
class B
{
public:
int b;
B() { cout << "B 생성자 호출" << endl; }
};
class C : public A, public B
{
public:
int c;
C() : A(), B() { cout << "C 생성자 호출" << endl;}
};
int main()
{
C c;
}
성공적으로 컴파일 하였다면
위 처럼 A -> B -> C 순으로 호출됨을 알 수 있습니다. 그렇다면 이번에는,
class C : public A, public B
에서
class C : public B, public A
로 바꾸고 컴파일을 해보세요. 재미있게도;
로 이번에는 B 의 생성자가 A 보다 먼저 호출됨을 알 수 있습니다. 몇 번 더 실험을 해보면 이 순서는 다른 것들에 의해 좌우되지 않고 오직 상속하는 순서에만 좌우 됨을 알 수 있습니다.
사실 다중 상속은 실제 프로그래밍에서 많이 쓰이지는 않습니다. 왜냐하면 다음과 같은 위험이 언제나 도사리고 있기 때문이지요.
class A
{
public:
int a;
};
class B
{
public:
int a;
};
class C : public B, public A
{
public:
int c;
};
위처럼 만일 두 개의 클래스에서 이름이 같은 멤버 변수나 함수가 있다고 해봅시다. 예를 들어 위 예에서는 클래스 A 와 B 에 모두 a 라는 이름의 멤버 변수가 들어가 있습니다.
int main()
{
C c;
c.a = 3;
}
그렇다면 만일 클래스 C 의 객체를 생성해서, 위 처럼 중복되는 멤버 변수에 접근한다면;
error C2385: ambiguous access of 'a'
1> could be the 'a' in base 'B'
1> or could be the 'a' in base 'A'
위 처럼 B 의 'a' 인지, A 의 'a' 인지 구분할 수 없다는 오류를 발생하게 됩니다. 마찬가지로, 클래스 A 와 B 에 같은 이름의 함수가 있다면 똑같이 어떤 함수를 호출해야 될 지 구분할 수 없겠지요. 그 외에도 다중 상속은 코드 구조를 매우 복잡하게 만드는 경향이 있기 때문에 C++ 이외에 많은 언어들 (자바, C# 등) 에서는 다중 상속 기능을 지원하고 있지 않습니다.
다중 상속의 또 다른 문제는 일명 '다이아몬드 상속(diamond inheritance)' 혹은 '공포의 다이아몬드 상속(dreadful diamond of derivation)' 이라고 부르는 형태의 다중 상속에 있습니다. 예를 들어 다음과 같은 형태의 상속 관계를 생각해봅시다.
class Human
{
// ...
};
class HandsomeHuman : public Human
{
// ...
};
class SmartHuman : public Human
{
// ...
};
class Me : public HandsomeHuman, public SmartHuman
{
// ...
};
일단 베이스 클래스로 Human 이라는 클래스가 있고, HandsomeHuman 과 SmartHuman 클래스는 Human 클래스를 모두 상속 받습니다. 그리고 두 가지 특성을 모두 보유한 나(Me) 라는 클래스는, HandsomeHuman 과 SmartHuman 클래스를 둘 다 상속 받습니다. 이를 그림으로 표현하자면 아래와 같은 다이아몬드 모양이 나오게 됩니다.
상속이 되는 두 개의 클래스가 공통의 베이스 클래스를 포함하고 있는 형태를 가리켜서 다이아몬드 상속이라고 부릅니다. 이러한 형태의 상속에 문제점은 보기에도 명백합니다. 만일 Human 에 name 이라는 멤버 변수가 있다고 해봅시다. 그러면 HandsomeHuman 과 SmartHuman 은 모두 Human 을 상속 받고 있으므로, 여기에도 name 이라는 변수가 들어가게 됩니다. 그런데 Me 가 이 두 개의 클래스를 상속 받으니 Me 에서는 name 이라는 변수가 겹치게 되는 것이지요. 결과적으로 볼 때 Handsome 과 SmartHuman 을 아무리 안겹치게 만든다고 해도, Human 의 모든 내용이 중복되는 문제가 발생하게 됩니다.
다행이도 이를 해결할 수 있는 방법이 있습니다.
class Human
{
public:
// ...
};
class HandsomeHuman : public virtual Human
{
// ...
};
class SmartHuman : public virtual Human
{
// ...
};
class Me : public HandsomeHuman, public SmartHuman
{
// ...
};
이러한 형태로 Human 을 virtual 로 상속 받는다면, Me 에서 다중 상속 시에도, 컴파일러가 언제나 Human 을 한 번만 포함하도록 지정할 수 있게 됩니다. 참고로, 가상 상속 시에, Me 의 생성자에서 HandsomeHuman 과 SmartHuman 의 생성자를 호출함은 당연하고, Human 의 생성자 또한 호출해주어야만 합니다.
앞에서도 이야기 하였지만 반드시 필요한 경우가 아니라면 다중 상속을 피하는 것이 좋습니다. 왜냐하면 다중 상속을 사용하게 되면 프로그램의 구조가 매우 복잡해질 뿐더러 예상치 못한 오류를 발생할 가능성이 매우 높기 때문이지요. 실제로 다중 상속을 이용해서 해결해야 될 것 같은 문제도 알고보면 단일 상속을 통해 해결할 수 있는 경우가 매우 많습니다 (어떤 사람들은 100% 라고 주장하기도 하지요) 다중 상속에 좀 더 자세히 알고 싶은 분들은 이 글을 읽는 것이 많은 도움이 될 것입니다. http://www.drdobbs.com/cpp/multiple-inheritance-considered-useful/184402074
아무래도 이번 강좌는 상속에 대한 중요한 요소들을 간단 하게 짚고 넘어가는 것이라 실질적인 프로그램은 만들지 않았습니다. 하지만, 가상 함수와 상속이 어떻게 돌아가는지 완벽히 이해하는 것이 좋습니다. 저의 경우, C++ 처음 배울 때, 이 부분에서 많이 헷갈려서 고생을 한 기억이 있습니다. 여러분들도 가상 함수를 포함하는 간단한 프로그램을 작성해서 어떻게 함수들이 호출되는지 살펴보시기 바랍니다.
강좌를 보다가 조금이라도 궁금한 것이나 이상한 점이 있다면 꼭 댓글을 남겨주시기 바랍니다. 그 외에도 강좌에 관련된 것이라면 어떠한 것도 질문해 주셔도 상관 없습니다. 생각해 볼 문제도 정 모르겠다면 댓글을 달아주세요.
현재 여러분이 보신 강좌는<<씹어먹는 C++ - <6 - 3. 가상함수와 상속에 관련한 잡다한 내용들>>> 입니다. 이번 강좌의 모든 예제들의 코드를 보지 않고 짤 수준까지 강좌를 읽어 보시기 전까지 다음 강좌로 넘어가지 말아주세요
출처: http://itguru.tistory.com/211 [Programming IT]
=================================
=================================
=================================
'프로그래밍 관련 > 언어들의 코딩들 C++ JAVA C# 등..' 카테고리의 다른 글
C/C++ 스레드 concurrent_vector, concurrent_queue, parallel_for, parallel_for_each 등등 관련 (0) | 2017.07.11 |
---|---|
C/C++ 랜덤숫자, 난수 생성 함수 rand, srand 사용법 및 중복 없는 난수 생성 관련 (1) | 2017.07.06 |
C/C++ 스레드, IOCP 관련 (0) | 2017.06.28 |
C, C++ 스레드(Thread) 관련 WaitForSingleObject 함수 (0) | 2017.06.15 |
[C++] string to int - string에서 int로 변환 (0) | 2017.06.14 |