기본 콘텐츠로 건너뛰기

scala 8. 상속

1. extends

상속은 자바와 마찬가지로
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 필드를 초기화 해준다.

댓글

이 블로그의 인기 게시물

메일서버가 스팸으로 취급받을때

설치한 메일서버를 통해 발송되는 메일이 스팸으로 들어가는 경우가 더러 있다. 이게 한번 들어가기는 쉬운데, 빠져나오기는 드럽게 힘든것 같다... 본인의 경우에는 우선 국내서비스에는 별 무리 없이 들어간다. (naver,daum 등) 그런데 해외메일 그중 Gmail, Hotmail 에는 에누리없이 스팸으로 간주되고 있었다. Gmail같은 경우에는 그래도 스팸함으로 발송은 제대로 되는반면에 Hotmail같은경우에는 아예 수신자체가 안되는 경우도 더러있다.. ㅡ,.ㅡ; 제일 좋은 방법은 Gmail,Hotmail에 전화걸어서 우리 메일서버 IP white Ip로 등록해달라!!! 하면 좋지만, 얘네들은 걸어봤자 자동응답기고, 문의채널은 구글 그룹스 게시판이 전부다.. 본론으로 들어가서. 해외 메일이 차단될 경우 내 매일서버ip가 스팸ip로 등록되 버린 경우일 수 있다. (본인처럼. ㅎ) 이것부터 조회 해보고 싶으면 RBL(real-time blocking List) 체크를 해야 하는데, RBL체크 해주는 사이트는 꽤 많이 있고, 그중 좀 깔끔해 보이는곳 하나 소개. http://www.anti-abuse.org/ 메일서버ip 입력하고 조회해보면 쭈루룩 리스트가 나온다. 그 중 빨간불이 들어온 부분이 메일 서버가 스팸서버가 된 각종 이유들이다.ㅋ 본인의 경우 CBL 때문에 걸렸는데, 내용은 아래와 같다. This IP address is HELO'ing as  "localhost.localdomain"  which violates the relevant standards (specifically: RFC5321). 메일서버 도메인에 별다른 작업을 안해놓아서 "localhost.localdomain" 으로 설정되어있었다. 만약 CBL만 바로 테스트 해보고 싶으면 http://cbl.abusea...

[javascript] 특정시간에만 함수 실행

특정시간에만 팝업을 띄우려면?? 특정시간에만 로그인을 막으려면?? 특정시간에만 할일은 의외로 참 많다. 방법? 딱히 없다. 현재시간 구해서 시작시간, 종료시간 사이에 있을때 시작하는 수밖엔. if ((현재시간 > 시작시간) && (현재시간 < 종료시간)){ .. 팝업노출(); 공사페이지 리다이렉트(); 기타등등(); .. } 자바스크립트로 작성하면 다음과 같다. var startdate = "2014012008" ; var enddate = "2014012418" ; var now = new Date (); //현재시간 year = now. getFullYear (); //현재시간 중 4자리 연도 month = now. getMonth () + 1 ; //현재시간 중 달. 달은 0부터 시작하기 때문에 +1 if ((month + "" ). length < 2 ){ month = "0" + month; //달의 숫자가 1자리면 앞에 0을 붙임. } date = now. getDate (); //현재 시간 중 날짜. if ((date + "" ). length < 2 ){ date = "0" + date; } hour = now. getHours (); //현재 시간 중 시간. if ((hour + "" ). length < 2 ){ hour = "0" + hour; } today = year + "" + month + "" + date + "" + hour; //오늘 날짜 완성. / / 시간비교 i...

스레드 동기화1 - syncronized

구현 스레드를 구현하는 방법은 2가지다. 1. Thread 클래스를 extends 한다. 2. Runnable 인터페이스를 implements 한다. 뭐 사실 Thread는 생성자의 변수로 Runnable을 취한다. public Thread (Runnable target) { init( null, target , "Thread-" + nextThreadNum () , 0 ) ; } 그리고 Runnable 인터페이스는 run() 이라는 단일함수를 갖는 인터페이스이다. @FunctionalInterface public interface Runnable { /** * When an object implementing interface <code> Runnable </code> is used * to create a thread, starting the thread causes the object's * <code> run </code> method to be called in that separately executing * thread. * <p> * The general contract of the method <code> run </code> is that it may * take any action whatsoever. * * @see java.lang.Thread#run() */ public abstract void run () ; } 그렇기 때문에 람다식으로 표현이 가능한 것이다. 동기화 동기화는 동시에 같은자원의 접근을 제한하고자 할때 사용한다. 예를들어, 한 우물에서 물을 15번 길어야 하는 일을 해야할때 5명이서 3번만 하면 수고를 5배로 줄일수 있다. 그런데 우물은 하난데 동시에...