c++ 생성자에 대하여
글쓴이: stella770 / 작성시간: 목, 2017/06/01 - 7:21오후
안녕하세요. 항상 수고가 많으십니다.
다름이 아니라 explicit 키워드를 보다가
혼자 테스트 해보던 도중
class MyClass
{
public:
MyClass(){}
MyClass(int num) : imember2(num){}
MyClass(char cnum) : cmember(cnum){}
MyClass(double dnum) : dmember(dnum){}
private:
int imember;
int imember2;
double dmember;
char cmember;
};
void main()
{
MyClass temp = 10; //가능 (생성자를 묵시적으로 호출) MyClass temp(10) 과 동일
MyClass test(); //디폴트 생성자 호출
test = 20; //정의한 자료형의 생성자를 타고 클래스의 일부변수가 초기화됨
test = 'z'; // 위와 동일
test = 4568.21; //위와 동일
}
객체를 생성하고 난 뒤 인데
객체에 정의한 타입에 맞는 생성자를 타는 이유는 무엇인가요?
저번에 올렸는데 답변이 없으셔서 다시 재질문합니다 ㅠ.ㅠ.
Forums:
컴파일이 안 될 것만 같은 코드로군요. 컴파일은 해
컴파일이 안 될 것만 같은 코드로군요. 컴파일은 해 보셨나요?
당연히 됩니다. IDE는 VS2013입니다.
문제는 경고메세지도 안뜬다는 겁니다.
값도 정확히 들어갑니다.
당연히 되는 게 아닙니다. 되는 게 이상한 거에요.
당연히 되는 게 아닙니다. 되는 게 이상한 거에요.
위 코드는 MyClass형 객체 test를 선언하는 게 아니라, 파라미터 없이 MyClass 객체를 반환하는 함수 test를 선언하는 것입니다.
그러니 이후 test를 MyClass 객체처럼 덮어쓰는 코드에서 문제가 생길 수밖에요.
혹시 몰라서 제 VS2017로 컴파일해보니 예상했던 컴파일러 에러 메시지가 뜨는군요.
VS2013에서 이게 컴파일이 된다면, 사실 컴파일이 되고 있는 게 아니거나 VS2013이 표준을 제대로 준수하지 않는 겁니다.
뭐, 별로 치명적인 문제는 아닙니다. 그냥 아래와 같이 바꿔주면 됩니다.
그 전에. 질문자님의 코드는 생성자를 직접 호출하는 게 아닙니다. 마치 함수 호출처럼 보인다고 해도 말이죠.
이에 대해서는 제가 예전에 답변을 한 번 쓴 적이 있는데 참고해보세요:
https://kldp.org/node/157183
본론으로 들어가면, 클래스는 프로그래머가 만들어주지 않은 경우에도 어떤 조건을 만족하면 소위 special member function이라는 것을 가지게 됩니다.
같은 클래스 타입의 객체로부터 대입을 하는 연산자가 그 중 하나입니다.
test에 대해 생성자가 다시 불리는 게 아닙니다. test는 이미 생성된 객체이고 생성자가 다시 불릴 일은 없습니다.
그런데 test에 20을 "대입"하는 방법이 없습니다. 그런 일을 하는 대입 연산자가 없으니까요. 컴파일러는 차선책을 찾으려 합니다.
int
(20)을 MyClass 타입으로 변환할 수 있고, 그렇게 하면 대입을 할 수 있게 됩니다. MyClass를 받는 대입 연산자는 (프로그래머가 작성한 것은 아니지만) 존재하니까요. 그래서 결국 위 코드는 아래와 같은 의미가 됩니다.이하 다른 코드들도 마찬가지입니다.
대입 연산자에 대해 더 알고 싶으시면 레퍼런스를 참조하세요:
http://en.cppreference.com/w/cpp/language/copy_assignment
C++11부터 "대입"은 "복사 대입"과 "이동 대입"으로 나뉘었습니다. 위 코드는 "이동 대입"됩니다.
http://en.cppreference.com/w/cpp/language/move_assignment
임시객체
test = 20; //정의한 자료형의 생성자를 타고 클래스의 일부변수가 초기화됨
test = 'z'; // 위와 동일
test = 4568.21; //위와 동일
20, 'z', 4578.21 이 MyClass type으로 casting되는 과정에서
MyClass type의 임시객체가 생성됩니다.
이 임시객체가 test에 대입되는 거고요.
댓글 달기