언어/JAVA
[JAVA] Abstract(추상클래스)
hvoon
2022. 9. 8. 11:15
-상속을 통한 다형성(부모클래스를 상속받아 여러형태의 자식클래스를 생성 & 활용 & type casting)을 구현함에 있어서 강제성을 부여하기 위한 클래스
-일반적인 상속의 문제점: 강제성의 부재
-상속을 통해 다형성(부모클래스를 상속받아 여러 형태의 자식 클래스를 생성하고 활용함)을 구현하고자 하여도, 하위클래스에서 메소드 오버라이딩을 구현하지 않으면 다형성을 완벽히 구현할 수 없음. 이때, 오버라이딩을 구현하지 않는것이 문법상 문제가 없다는 점을 강제성의 부재라고 함
class Animal{
public void crying() {
System.out.println("울음 소리~");
}
}
class Dog extends Animal{
public void crying() {
System.out.println("멍~멍~");
}
} // 부모클래스의 crying 메소드를 오버라이딩한 강아지 클래스
class Cat extends Animal{
}// 부모클래스의 crying 메소드를 오버라이딩 하지 않은 고양이 클래스
public class Abstract01 {
public static void main(String[] args) {
Dog dog = new Dog();
dog.crying();
Cat cat = new Cat();
cat.crying();
}
}
추상메소드를 포함한 클래스: 추상 메서드를 하나이상 포함하면 추상클래스라고 부름
추상메서드: 메소드의 원형만 존재하고 실제 내용이 없는 메소드
추상메소드의 작성방법
접근지정자 abstract 리턴값의타입 메소드명( 매개변수 );
- 추상 클래스는 일반 클래스와 동일하게 일반 멤버변수 일반 메소드, 생성자 등을 포함할 수 있음.
- 단, 추상 메소드를 포함할 수 있는 특징이 추가된 클래스
- 추상클래스는 상속을 통한 다형성 구현을 위해서 생성됨
abstract class AbstractAnimal{ // 추상클래스
public abstract void crying(); // 추상메서드
}
class DogA extends AbstractAnimal{
@Override
public void crying() {
System.out.println("멍~멍~");
}
}
class CatA extends AbstractAnimal{
@Override
public void crying() {
System.out.println("야옹~");
}
}
public class Abstract02 {
public static void main(String[] args) {
DogA d = new DogA();
d.crying();
CatA c = new CatA();
c.crying();
// 추상클래스는 상속전용으로 만들어지므로, 아래와 같이 단독으로 객체 생성을 하지 못함
// AbstractAnimal a = new AbstractAnimal(); // 에러
// 추상메서드를 포함하고 있는 클래스는, 말그대로 완전하지 않은 클래스상태라는 뜻
// 따라서 자체적인 객체 생성이 되지 않음.
// 추상클래스로 추상클래스만의 객체를 만들수는 없지만
// 추상클래스(부모)의 레퍼런스 변수로, 자식클래스의 인스턴스 주소는 저장 가능
// AbstractAnimal a = new AbstractAnimal(); X
// AbstractAnimal b = new DogA(); O
// AbstractAnimal c = new CatA(); O
AbstractAnimal dog = new DogA();
AbstractAnimal cat = new CatA();
dog.crying();
cat.crying();
dog.addAge();
cat.addAge();
increamentAge(d);
increamentAge(c);
}
// 자식 클래스의 인스턴스를 전달인수로 보내면,
// 함수의 매개변수인 부모 레퍼런스가 받아서 사용할 수 있음
public static void increamentAge( AbstractAnimal animal ) {
animal.addAge();
}
}
추상 클래스의 단점
- 추상 메소드 구현의 강제성이 아래의 경우 단점이 될 수 있음
- 추상 메소드의 갯수가 많으면 상속에 부담을 주게 됨
- 자식 클래스에서 사용하지 않을 추상 메소드라도, 객체 생성을 위해 반드시 구현해야 하는 문제점이 있음
abstract class AbstractA{
public abstract void test1();
public abstract void test2();
public abstract void test3();
public abstract void test4();
public abstract void test5();
public abstract void test6();
public abstract void test7();
}
class AbstractA_Sub1 extends AbstractA{
@Override
public void test1() {}
@Override
public void test2() { }
@Override
public void test3() { }
@Override
public void test4() { }
@Override
public void test5() { }
@Override
public void test6() { }
@Override
public void test7() { }
}
// AbstractA 클래스를 상속 받은 클래스는 필요없는 메서드가 있어도, 추상매서드를 모두 오버라이드(구현)해야함
// 모든 추상매서드가 구현(오버라이드)된 AbstractA_Adapter 클래스를 생성하고,
// 이 클래스를 상속받아 사용하면 필요없는 매서드를 강제로 구현(오버라이드)하지 않고
// 필요한 것만 구현하여 사용 가능
class AbstractA_Adapter extends AbstractA {
@Override
public void test1() { }
@Override
public void test2() { }
@Override
public void test3() { }
@Override
public void test4() { }
@Override
public void test5() { }
@Override
public void test6() { }
@Override
public void test7() { }
}
class AbstractA_Sub2 extends AbstractA_Adapter {
public void test1() {
System.out.println("test1 오버라이딩~!");
}
}
public class Abstract03 {
public static void main(String[] args) {
// 어댑터 클래스를 상속받은 클래스도 추상클래스의 자식(손자)클래스가 됨
AbstractA obj = new AbstractA_Sub2();
obj.test1();
// 할아버지 손자간 다형성을 구현하여 물려주고 오버라이딩된 메서드를 할아버지 레퍼런스가 사용 가능
// 함수의 전달인수로 손자인스턴스 주소를 보내고,
// 함수의 매개변수로 할아버지 레퍼런스 사용이 아래와 같이 가능함
test( obj );
}
public static void test( AbstractA object1 ) {
}
}