생성자 선언과 호출
new 연산자는 객체를 생성한 후 연이어 생성자를 호출해서 객체를 초기화하는 역할을 한다. 객체 초기화란 필드 초기화를 하거나 메서드를 호출해서 객체를 사용할 준비를 하는 것을 말한다.
클래스 변수 = new 클래스(); // 클래스() = 생성자 호출
생성자가 성공적으로 실행이 끝나면 new 연산자는 객체의 주소를 리턴한다. 리턴된 주소는 클래스 변수에 대입되어 객체의 필드나 메서드에 접근할 때 이용된다.
기본 생성자
모든 클래스는 생성자가 존재하며, 하나 이상을 가질 수 있다. 클래스에 생성자 선언이 없으면 컴파일러는 다음과 같은 기본 생성자(Default Constructor)를 바이트코드 파일에 자동으로 추가한다.
[public] 클래스() {}
클래스가 public으로 선언되면 기본 생성자도 public이 붙지만, 클래스가 public 없이 class로만 선언되면 기본 생성자에도 public이 붙지 않는다.
생성자 선언
객체를 다양하게 초기화하기 위해 개발자는 생성자를 다음과 같이 직접 선언할 수 있다.
클래스(매개변수, ...) {
// 객체의 초기화 코드
}
생성자는 메서드와 비슷한 모양을 가지고 있으나. 리턴 타입이 없고 클래스 이름과 동일하다는 차이점이 있다. 매개 변수는 new 연산자로 생성자를 호출할 때 매개값을 생성자 블록 내부로 전달하는 역할을 한다. 다음과 같이 Car 생성자를 호출할 때 3개의 매개값을 블록 내부로 전달하고 가정해 보자.
Car myCar = new Car("그랜저", "검정", 300);
3개의 매개값을 순서대로 매개변수로 대입받기 위해서는 다음과 같이 생성자가 선언되어야 한다.
public class Car {
Car(String model, String color, int maxSpeed) {...}
}
필드 초기화
객체마다 동일한 값을 가지고 있다면 필드 선언 시 초기값을 대입하는 것이 좋고, 객체마다 다른 값을 가져야 한다면 생성자에서 필드를 초기화하는 것이 좋다. 예를 들어 Korean 클래스를 선언한다고 가정해 보자. 한국인이므로 nation(국가)은 대한민국으로 동일한 값을 가지지만, name(이름)과 ssn(주민등록번호)은 한국인마다 다르므로 생성자에서 초기화하는 것이 좋다.
public class Korean {
String nation = "대한민국";
String name;
String ssn;
public Korean(String n, String s) {
name = n;
ssn = s;
}
}
생성자의 매개값은 new 연산자로 생성자를 호출할 때 주어진다. k1과 k2가 참조하는 객체는 주어진 매개값으로, name과 ssn 필드가 각각 초기화 된다.
Korean k1 = new Korean("박자바", "123456-1234567");
Korean k2 = new Korean("김자바", "123456-1234567");
매개변수 이름이 너무 짧으면 코드 가독성이 좋지 않기 때문에 가능하면 초기화시킬 필드명과 동일한 이름을 사용하는 것이 좋다.
public Korean(String name, String ssn) {
this.name = name;
this.ssn = ssn;
}
위와 같은 경우에는 매개변수명이 필드명과 동일하기 때문에 필드임을 구분하기 위해 this 키워드를 필드명 앞에 붙여주었다. 생성자 안에서 선언한 매개변수 이름과 필드에서 선언한 변수의 이름이 동일할 때, 같은 변수를 언급한다면 생성자가 더 안쪽에 있는 코드이기 때문에 매개변수로 인식한다. this.변수명을 언급하면 필드에서 선언한 변수로 인식한다.
생성자 오버로딩
매개값으로 객체의 필드를 다양하게 초기화하려면 생성자 오버로딩(Overloading)이 필요하다. 생성자 오버로딩이란 매개변수를 달리하는 생성자를 여러 개 선언하는 것을 말한다.
public class Car {
car() {...}
car(String model) {...}
car(String model, String color) {...}
car(String model, String color, int maxSpeed) {...}
}
매개변수의 타입과 개수 그리고 선언된 순서가 똑같을 경우 매개변수 이름만 바꾸는 것은 생성자 오버로딩이 아니다.
Car(String model, String color) {...}
Car(String color, String model) {...} // 오버로딩 아님, 컴파일 에러
생성자가 오버로딩되어 있을 경우, new 연산자로 생성자를 호출할 때 제공되는 매개값의 타입과 수에 따라 실행될 생성자가 결정된다.
Car car1 = new Car();
Car car2 = new Car("그랜져");
Car car3 = new Car("그랜져", "흰색");
다른 생성자 호출
생성자 오버로딩이 많아질 경우 생성자 간의 중복된 코드가 발생할 수 있다. 매개변수의 수만 달리하고 필드 초기화 내용이 비슷한 생성자에서 이러한 중복 코드를 많이 볼 수 있다. 이 경우 공통 코드를 한 생성자에게만 집중적으로 작성하고, 나머지 생성자는 this(...)를 사용하여 공통 코드를 가지고 있는 생성자를 호출하는 방법으로 개선할 수 있다.
Car(String model) {
this(model, "은색", 250);
}
Car(String model, String color) {
this(model, color, 250);
}
// 공통 초기화 코드
Car(String model, String color, int maxSpeed) {
this.model = model;
this.color = color;
this.maxSpeed = maxSpeed;
}
this(매개값,...)는 생성자의 첫 줄에 작성되며 다른 생성자를 호출하는 역할을 한다. 호출하고 싶은 생성자의 매개변수에 맞게 매개값을 제공하면 된다. this() 다음에는 추가적인 실행문을 작성할 수 있는데, 호출되는 생성자의 실행이 끝나면 원래 생성자로 돌아와서 다음 실행문을 실행한다.
Car(String model) {
this(model, "은색", 250);
// 추가적인 실행문
}
Car(string model, String color, int maxSpeed) {
this.model = model;
this.color = color;
this.maxSpeed = maxSpeed;
}
'Language > JAVA' 카테고리의 다른 글
| [JAVA] 인스턴스 멤버와 정적 멤버 (0) | 2024.09.04 |
|---|---|
| [JAVA] 메서드 (0) | 2024.09.04 |
| [JAVA] 필드 (0) | 2024.09.04 |
| [JAVA] 클래스 (0) | 2024.09.04 |
| [JAVA] 객체지향 프로그래밍 (3) | 2024.09.04 |