기본 콘텐츠로 건너뛰기

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

[linux] 모든 cron 확인

예전에 만들어놓은 cron이 요새 안돌고 있어서 찾아봤다. 물론 내가 만들었던 아니고ㅋ crontab -l no crontab for root 헐, 내가 찾던 cron이 안보인다. 분명히 예전 사수가 여기다 작업 해놓은것 같았는데... cron이 뭐하는 놈인지는 알아도, 어떻게 생겨먹은놈인지는 몰랐다ㅋ 이리저리 폴더 뒤져서 그때 스크립트 파일을 찾아보기 시작했다. 멍청한짓을 한 참 한후에 혹하는 마음에 로컬 사용자 뭐있나 찍어봄 cat /etc/passwd 찍어보믄 계정별로 결과가 쭈루루룩 나오는데, 뜻은 이거다. 계정:암호:UID:GID:정보:디렉토리:쉘 passwd 파일 설명은 ( http://www.linuweb.com/?p=248 ) 여기 님 블로그에 잘 되어있음. 암튼 결국 계정별로 cron 걸려있는거 죄다 찍어보니 for user in $(cut -f1 -d: /etc/passwd); do echo id $user; crontab -u $user -l ; done 한번에 다찍어주는 명령어는 따로 없나보다. 간단하게 스크립트로 찍어주는 수밖에. 결국 내가 찾던 cron은 처음보는 계정에 들어있었음. 그리고 cron이 안돌던 이유는 ftp가 막혀서였고 ㅋ