1. extends
상속은 자바와 마찬가지로
extends 라는 키워드를 사용
상속을 막고 싶으면 final을 사용
2. override
오버라이드(함수를 상속받아 재정의, cf 오버로드-함수의 파라미터 갯수를 달리하여 재정의)를 할경우 추상메소드인경우에는 그냥 진행하면 되지만, 그렇지 않은경우 메소드 앞에 override를 붙여준다.
3. 슈퍼클래스 생성자
- java에서 부모클래스의 생성자를 이용할때 super() 사용한다.
- 부모클래스의 생성자의 호출은 클래스의 상속관계를 거슬러 반복되며, 최고조상인 Object클래스의 생성자인 Object()까지 가서 끝이난다.
- 만약 모든클래스에서 생성자를 만들지 않으면, 컴파일러가 생성자의 첫줄에 super(); 를 자동으로 추가한다고 한다. (참고 java-에서-super를-사용하는-이유 http://goo.gl/uC0Ptm)
5장을 다시 복기하면,
scala에서는 하나의 기본생성자와 여러개의 보조생성자가 있다.
보조생성자는 앞선 보조생성자나 기본생성자 호출로 시작해야 한다.
고로, 보조생성자는 부모클래스 생성자를 직접 호출하지 않는다.
따라서 자식클래스의 기본생성자가 부모클래스의 생성자를 호출한다.
4. 필드오버라이드
스칼라에서 필드는 비공개필드와 접근자/변경자 메소드로 구성
val(혹은 인자없는 def)은 같은 이름의 또 다른 val 필드로 오버라이드 가능.
자식클래스는 비공개필드, 공개게터 슈퍼클래스 게터(혹은 메소드)를 오버라이드하는 게터를 가짐.
실행
제약사항 (오버라이드 가능한경우)
- def는 다른 def만
- val은 다른 val 또는 인자없는 def만
- var은 추상 클래스의 var만
5. 익명 서브클래스
java의 익명클래스와 동일한듯하다. (java 익명클래스 - http://happystory.tistory.com/64 )
일회용으로 클래스를 변형해서 사용하고자 할때 사용한다. 일반적으로 자주 쓸일이 있겠냐만은 안드로이드와 같은 gui 환경에서 버튼 이벤트리스너를 커스터마이징할때 많이 사용한다.
palyframework application.object ( '/', root 페이지 접근시 매핑되는 controller)
6 추상클래스
메소드정의, 멤버필드 초기화가 안되어있는클래스
자바와 마찬가지로
abstract 키워드를 사용
7. 생성순서와 조기정의
Creature를 상속받은 Ant라는 클래스가 있다. range를 오버라이드 했을때,
Ant 클래스의 env.length는 얼마일까,
결과는 '0' 이다.
원리는 이렇다.
1. Ant 생성자는 자신의 생성에 앞서 Creature 생성자를 호출
2. Creature 생성자는 range의 필드값을 10으로 설정
3. Creature 생성자는 env 배열을 초기화하기 위해 range() 게터를 호출
4. 이메소드는 (아직 초기화되지않은) Ant 클래스의 range 필드를 주도록 오버라이드되었음
5. range 메소드는 0을리턴(오브젝트가 할당되었을때 가지는 초기값)
6. env의 길이는 0
7. Ant 생성자는 range 필드를 0으로 설정
이럴때는 "조기정의" 문법으로 해결
상속은 자바와 마찬가지로
extends 라는 키워드를 사용
class Person { var name = "" override def toString = getClass().getName + "[name=" + name + "]" } ... class Employee extends Person{ var salary = 0.0 override def toString = super.toString + "[salary="+ salary + "]" }
상속을 막고 싶으면 final을 사용
2. override
오버라이드(함수를 상속받아 재정의, cf 오버로드-함수의 파라미터 갯수를 달리하여 재정의)를 할경우 추상메소드인경우에는 그냥 진행하면 되지만, 그렇지 않은경우 메소드 앞에 override를 붙여준다.
class Person { var name = "" override def toString = getClass().getName + "[name=" + name + "]" } ... class Employee extends Person{ var salary = 0.0 override def toString = super.toString + "[salary="+ salary + "]" }
실행
object test { def main(args: Array[String]): Unit ={ var ch = new Employee println(ch.toString) } } <result> [name=][salary=0.0]
3. 슈퍼클래스 생성자
- java에서 부모클래스의 생성자를 이용할때 super() 사용한다.
- 부모클래스의 생성자의 호출은 클래스의 상속관계를 거슬러 반복되며, 최고조상인 Object클래스의 생성자인 Object()까지 가서 끝이난다.
- 만약 모든클래스에서 생성자를 만들지 않으면, 컴파일러가 생성자의 첫줄에 super(); 를 자동으로 추가한다고 한다. (참고 java-에서-super를-사용하는-이유 http://goo.gl/uC0Ptm)
5장을 다시 복기하면,
scala에서는 하나의 기본생성자와 여러개의 보조생성자가 있다.
보조생성자는 앞선 보조생성자나 기본생성자 호출로 시작해야 한다.
고로, 보조생성자는 부모클래스 생성자를 직접 호출하지 않는다.
따라서 자식클래스의 기본생성자가 부모클래스의 생성자를 호출한다.
class Person (name: String){ override def toString = "[name=" + name + "]" }
class Employee(name: String, age: Int, val salary: Double) extends Person(name){ override def toString = super.toString + "[salary="+ salary + "won]" + "[age=" + age + "years old]" }실행
def main(args: Array[String]): Unit ={ println() var ch = new Employee("changpd",10,100) println(ch.toString) } <result> [name=changpd][salary=100.0won][age=10years old]
4. 필드오버라이드
스칼라에서 필드는 비공개필드와 접근자/변경자 메소드로 구성
val(혹은 인자없는 def)은 같은 이름의 또 다른 val 필드로 오버라이드 가능.
자식클래스는 비공개필드, 공개게터 슈퍼클래스 게터(혹은 메소드)를 오버라이드하는 게터를 가짐.
class Person (val name: String){ override def toString = "[name=" + name + "]" } class SecretAgent(codename: String) extends Person(codename) { override val name = "secret" override val toString = "secret" } class Agent (name:String) extends Person(name){ }
실행
def main(args: Array[String]): Unit ={ var p1 = new Person("p1") println(p1.toString) var s1 = new SecretAgent("s1") println(s1.toString) var a1 = new Agent("a1") println(a1.toString) } <result> [name=p1] secret [name=a1]
제약사항 (오버라이드 가능한경우)
- def는 다른 def만
- val은 다른 val 또는 인자없는 def만
- var은 추상 클래스의 var만
5. 익명 서브클래스
java의 익명클래스와 동일한듯하다. (java 익명클래스 - http://happystory.tistory.com/64 )
일회용으로 클래스를 변형해서 사용하고자 할때 사용한다. 일반적으로 자주 쓸일이 있겠냐만은 안드로이드와 같은 gui 환경에서 버튼 이벤트리스너를 커스터마이징할때 많이 사용한다.
def main(args: Array[String]): Unit ={ val alien = new Person("Fred"){ def greethig = "Greetings, Earthling! My name is " + name + "." } println(alien.greethig) } <result> Greetings, Earthling! My name is Fred.
palyframework application.object ( '/', root 페이지 접근시 매핑되는 controller)
object Application extends Controller { def index = Action { Ok(views.html.index("Your new application is ready.")) } }
6 추상클래스
메소드정의, 멤버필드 초기화가 안되어있는클래스
자바와 마찬가지로
abstract 키워드를 사용
7. 생성순서와 조기정의
class Creature { val range = 10 val env = new Array[Int](range) } class Ant extends Creature{ override val range = 2 }
Creature를 상속받은 Ant라는 클래스가 있다. range를 오버라이드 했을때,
Ant 클래스의 env.length는 얼마일까,
def main(args: Array[String]): Unit ={ var c1 = new Creature println("creature range" + c1.range) println("creature env" + c1.env.length) var a1 = new Ant println("ant range" + a1.range) println("ant env" + a1.env.length) } <result> creature range10 creature env10 ant range2 ant env0
결과는 '0' 이다.
원리는 이렇다.
1. Ant 생성자는 자신의 생성에 앞서 Creature 생성자를 호출
2. Creature 생성자는 range의 필드값을 10으로 설정
3. Creature 생성자는 env 배열을 초기화하기 위해 range() 게터를 호출
4. 이메소드는 (아직 초기화되지않은) Ant 클래스의 range 필드를 주도록 오버라이드되었음
5. range 메소드는 0을리턴(오브젝트가 할당되었을때 가지는 초기값)
6. env의 길이는 0
7. Ant 생성자는 range 필드를 0으로 설정
이럴때는 "조기정의" 문법으로 해결
class Bug extends { override val range =3 }with Creature
<result> bug range3 bug env3슈퍼클래스의 생성자를 호출하기전에 val 필드를 초기화 해준다.
댓글
댓글 쓰기