기본 콘텐츠로 건너뛰기

[JAVA] 참조 자료형의 형변환

 오늘은 참조자료형의 형변환에 대해서 정리해 보겠다.

사실 블로그 정리 전까지 아리송 까리송 했는데, 이제야 개념 확립이 된거 같아 정리.....

다음 코드를 보자

1
2
3
4
5
6
7
8
package c;
public class ParentCasting {
    public ParentCasting() {}
    public ParentCasting(String name){}
    public void printName(){
        System.out.println("print Parent!!!");
    }
}
cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
package c;
 
public class ChildCasting extends ParentCasting {
    public ChildCasting(){}
    public ChildCasting(String name){}
    public void printName() {
        System.out.println("print Child!!");
    }
 
    public void printHobby(String hobby) {
        System.out.println("My hobby is "+hobby);
    }
}
 
cs
위와 같이 부모와 자식의 객체가 상속 받고 있다고 하자.

다음과 같이 일반적으로 쓸 수 있을 것이다.

ParentCasting parent = new ParentCasting();
ChildCasting child = new ChildCasting();

위와 같이 하면 컴파일에 아무 지장이 없다. 그럼 어디서 많이 봐왔던? 다음 코드

ParentCasting obj = new ChildCasting();
 위 코드는 상속 관계가 성립하고, 자식의 생성자로 부모 인스턴스를 만들 수 있다. 
책 어디서 봐 왔던 형 변환이 필요 없는 방법이다.
(부모 입장에서 호출할 수 있는 메서드는 자식에서도 사용할 수 있으므로)

하지만 다음 코드는 에러를 내 뿜는다.
ChildCasting obj2 = new ParentCasting();
자식은 확장된 코드가 필요한데 좁아터진 부모 객체를 대입하려 했기 때문.
책 등에서는 위와 같이 표현을 하곤 하는데 나는 잘 이해가 가지 않았다.
그래서 메모리 구조를 그려 보았다.
메모리 구조인데 간단하게 설명하자면 스택엔 클래스의 인스턴스 정보가 힙에는 클래스의 객체 정보가 들어간다고 생각하면 된다.
위에서 보는 것처럼 obj2는 ParentCasting 객체를 가지고 있다. ParentCasting은 printName() 메서드 하나만 가지고 있고, ChildCasting은 printName()과 printHobby() 메서드 2개를 가지고 있으니 ChildCasting 입장에선 불만?인게 되어 에러가 발생하는 것이다. 

그럼 형변환 해주면 되지 않나??
ParentCasting obj = (ChildCasting)parent;

위와 같이 하면 컴파일 에러는 안난다. 근데, 실행시 예외가 발생한다.

근데 예외가 안날 수도 있다.(장난하늬??)
자식 객체에서 부모 객체를 그대로 쓰면(확장을 안하면) 위와 같이 쓸 수 있다. 

아니 그럼 어떻게 해야 하늬??
다음과 같이 하면 에러 없이 잘 수행 된다. 

ChildCasting child = new ChildCasting();
ParentCasting parent = child;
ChildCasting child2 = (ChildCasting)parent;

무엇이 다른지 유심히 보자. 위의 parent객체는 겉모습은 ParnetCasting이지만 알맹이는 ChildCasting이니 형변환해도 문제가 없다.(이해안가면 메모리구조를...)

여기 까지 형변환에 대해서 알아 보았다. 메모리 구조를 그려가며 생각하니 이제서야 이해가 가는것 같다. 

그럼 끝~!

댓글

이 블로그의 인기 게시물

[자바 웹 프로그래밍]2장 문자열 계산기 구현을 통한 테스트와 리펙토링

이번엔 2장에 나와 있는 내용 정리와 느낀점을 정리 해 보겠다. 1. main() 메소드를 활용한 테스트의 문제점.   - 소스코드 구현 후 정상적으로 동작하는지 확인 위해 일반적인 방법은 main()메소드를 활용하는 것이다.   - 실제 서비스를 담당하는 프로덕션 코드와 이 프로덕션 코드가 정상 동작 하는지 확인을 위한 main() 으로 나뉜다.   - 이 방법의 첫번째 문제점은 프로덕션코드와 main() 메서드가 함께 있다는 것이다.   - 프로덕션 코드와 테스트코드(main)을 분리할 수 있다.   - 두 번째 문제는 내가 구현하고 있는 메서드만 집중 할 수 없고, 클래스가 가지고 있는 모든 메서드를 테스트 할 수 밖에 없다.   - 다른 문제는 항상 콘솔로 확인을 할 수 밖에 없다는 것이다.   - 이를 위해 등장한 라이브러리가 JUnit 이다. 내 관심을 가지는 메서드에 대해 테스트 가능하다. 2. JUnit을 활용해 main() 메서드 문제 극복 2.1 한 번에 메서드 하나에만 집중.   - JUnit관련 라이브러리 추가 후  테스트 메서드에 @Test를 붙이면 된다.   - test 관련 코드 작성 후 Run > Run as> JunitTest를 실행해 보자.   - 각각 테스트 메서드를 독립적으로 실행할 수 있기 때문에 현재 내가 구현하고 있는 프로덕션 코드의 메서드에 집중할 수 있다. import org.junit.Test; public class CalculatorTest { @Test public void add() { Calculator cal = new Calculator(); System.out.println(cal.add(1,2)); } } 2.2 결과 값을 눈이 아닌 프로그램을 통해 자동화 import org.junit.Test; import static org.junit.Assert.assertEquals;

[고량주] 라오왕 연태고량주 플러스

나에게 처음 고량주란 이런것이다 라는걸 알려준 녀석이다. 부모님이 중국집을 하다 보니 가끔 초록색병 고량주를 먹었을때  역한 공업용 알콜 맛에 고량주는 나랑 안맞는다 생각했다가 우연히 양고기에 이녀석을 접한 뒤로 고량주의 맛을 알아버렸다... 제품명 : 라오왕 연태고량주플러스 제품유형 : 일반증류주 도수 : 34.2% 가격 : 9000원(홈플러스 익스프레스 기준) 재구매 의사 : 있다 시음평 : 역시 고량주 특유의 향인데, 열대과일 향도나고, 배향, 살짝 달달한 향이 난다.            목넘김은 34.2%에도 불구하고 그리 힘들지 않았다(주당이 된걸수도..)             중국요리나 양꼬치집에서 맛있는 술이 땡긴다면 강력추천한다.