반응형
Jake Seo
제이크서 위키 블로그
Jake Seo
전체 방문자
오늘
어제
  • 분류 전체보기 (715)
    • 일상, 일기 (0)
    • 백준 문제풀이 (1)
    • 릿코드 문제풀이 (2)
    • 알고리즘 이론 (10)
      • 기본 이론 (2)
      • 배열과 문자열 (8)
    • 데이터베이스 (15)
      • Planet Scale (1)
      • MSSQL (9)
      • 디비 기본 개념 (1)
      • SQLite 직접 만들어보기 (4)
    • 보안 (7)
    • 설계 (1)
    • 네트워크 (17)
      • HTTP (9)
      • OSI Layers (5)
    • 회고 (31)
      • 연간 회고 (2)
      • 주간 회고 (29)
    • 인프라 (52)
      • 도커 (12)
      • AWS (9)
      • 용어 (21)
      • 웹 성능 (1)
      • 대규모 서비스를 지탱하는 기술 (9)
    • 깃 (7)
    • 빌드 도구 (7)
      • 메이븐 (6)
      • 그레이들 (0)
    • Java (135)
      • 이펙티브 자바 (73)
      • 자바 API (4)
      • 자바 잡지식 (30)
      • 자바 디자인 패턴 (21)
      • 톰캣 (Tomcat) (7)
    • 프레임워크 (64)
      • next.js (14)
      • 스프링 프레임워크 (28)
      • 토비의 스프링 (6)
      • 스프링 부트 (3)
      • JPA (Java Persistence API) (5)
      • Nest.js (8)
    • 프론트엔드 (48)
      • 다크모드 (1)
      • 노드 패키지 관리 매니저 (3)
      • CSS (19)
      • Web API (11)
      • tailwind-css (1)
      • React (5)
      • React 새 공식문서 요약 (1)
      • HTML (Markup Language) (5)
    • 자바스크립트 (108)
      • 모던 자바스크립트 (31)
      • 개념 (31)
      • 정규표현식 (5)
      • 코드 스니펫 (1)
      • 라이브러리 (6)
      • 인터뷰 (24)
      • 웹개발자를 위한 자바스크립트의 모든 것 (6)
      • 팁 (2)
    • Typescript (49)
    • 리눅스와 유닉스 (10)
    • Computer Science (1)
      • Compiler (1)
    • IDE (3)
      • VSCODE (1)
      • IntelliJ (2)
    • 세미나 & 컨퍼런스 (1)
    • 용어 (개발용어) (16)
      • 함수형 프로그래밍 용어들 (1)
    • ORM (2)
      • Prisma (2)
    • NODEJS (2)
    • cypress (1)
    • 리액트 네이티브 (React Native) (31)
    • 러스트 (Rust) (15)
    • 코틀린 (Kotlin) (4)
      • 자바에서 코틀린으로 (4)
    • 정규표현식 (3)
    • 구글 애널리틱스 (GA) (1)
    • SEO (2)
    • UML (2)
    • 맛탐험 (2)
    • 리팩토링 (1)
    • 서평 (2)
    • 소프트웨어 공학 (18)
      • 테스팅 (16)
      • 개발 프로세스 (1)
    • 교육학 (1)
    • 삶의 지혜, 통찰 (1)
    • Chat GPT (2)
    • 쉘스크립트 (1)
    • 컴파일 (2)
    • Dart (12)
    • 코드팩토리의 플러터 프로그래밍 (4)
    • 플러터 (17)
    • 안드로이드 스튜디오 (1)
    • 윈도우즈 (1)
    • 잡다한 백엔드 지식 (1)
    • 디자인 패턴 (1)

블로그 메뉴

  • 홈
  • 태그
  • 방명록

공지사항

인기 글

태그

  • 빈 검증
  • 참조 해제
  • 알고리즘
  • 싱글톤
  • 느린 쿼리
  • 자바 검증
  • 스프링 검증
  • 러스트
  • 이펙티브 자바 item9
  • 디자인패턴
  • 자바스크립트 인터뷰
  • 자바스크립트
  • 객체복사
  • 자바
  • 플라이웨이트패턴
  • NEXT JS
  • 이펙티브자바
  • 서버리스 컴퓨팅
  • 슬로우 쿼리
  • pnpm
  • Pre-rendering
  • Java
  • Javadoc 자바독 자바주석 주석 Comment
  • 싱글턴
  • 자바 디자인패턴
  • prerendering
  • 추상 팩터리 패턴
  • item9
  • 싱글톤 패턴
  • 메이븐 페이즈
  • 메이븐 골
  • 작업기억공간
  • item8
  • 자바스크립트 면접
  • 외래키 제약조건
  • item7
  • try-with-resources
  • 프로그래머의 뇌
  • bean Validation
  • Next.js
  • 팩터리 메서드 패턴
  • next js app
  • MSSQL
  • 이펙티브 자바
  • serverless computing
  • 자료구조
  • 토비의 스프링
  • 메이븐 라이프사이클
  • 도커공식문서
  • rust

최근 댓글

최근 글

티스토리

hELLO · Designed By 정상우.
Jake Seo

제이크서 위키 블로그

코틀린 (Kotlin)/자바에서 코틀린으로

자바에서 코틀린으로 3장 - 자바클래스에서 코틀린 클래스로 요약

2022. 11. 30. 09:54

요약

생성자의 형태가 다르다.

  • 자바 버전
public class EmailAddress {
    private final String localPart;
    private final String domain;

    public EmailAddress(String localPart, String domain) {
        this.localPart = localPart;
        this.domain = domain;
    }

    public String getLocalPart() {
        return localPart;
    }

    public String getDomain() {
        return domain;
    }

    public static EmailAddress parse(String value) {
        var atIndex = value.lastIndexOf('@');
        if (atIndex < 1 || atIndex == value.length() - 1) {
            throw new IllegalArgumentException(
                    "EmailAddress must be two parts separated by @"
            );
        }

        return new EmailAddress(
                value.substring(0, atIndex),
                value.substring(atIndex + 1)
        );
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        EmailAddress that = (EmailAddress) o;
        return Objects.equals(localPart, that.localPart) && Objects.equals(domain, that.domain);
    }

    @Override
    public int hashCode() {
        return Objects.hash(localPart, domain);
    }

    @Override
    public String toString() {
        return localPart + "@" + domain;
    }
}
  • 코틀린 버전
class EmailAddress(val localPart: String, val domain: String) {
    override fun equals(o: Any?): Boolean {
        if (this === o) return true
        if (o == null || javaClass != o.javaClass) return false
        val that = o as EmailAddress
        return localPart == that.localPart && domain == that.domain
    }

    override fun hashCode(): Int {
        return Objects.hash(localPart, domain)
    }

    override fun toString(): String {
        return "$localPart@$domain"
    }

    companion object {
        fun parse(value: String): EmailAddress {
            val atIndex = value.lastIndexOf('@')
            require(!(atIndex < 1 || atIndex == value.length - 1)) { "EmailAddress must be two parts separated by @" }
            return EmailAddress(
                value.substring(0, atIndex),
                value.substring(atIndex + 1)
            )
        }
    }
}
  • 프로퍼티, 생성자, Getter 가 사라졌다.
    • 프로퍼티는 클래스 이름 옆에 val localPart 와 같은 형식으로 정의된다.
    • 생성자는 private constructor 키워드를 입력하지 않는 한 자동으로 공개 생성자가 만들어진다.
    • Getter 는 프로퍼티 선언 시 private val localPart 와 같이 선언하지 않는다면, 자동으로 만들어진다.
      • 다만 .getXxx() 의 형태로 사용하지 않고, 단순히 emailAddress.localPart 와 같은 형태로 사용한다.
  • 정적 메서드는 companion object 라는 곳 내부로 들어갔다.

기존의 자바의 관습에 익숙한 자바 개발자라면, 코틀린의 클래스 구문에 금방 적응할 수 있을 것이다. 짧은데도 가독성을 해치지 않고 무슨 일을 하는지도 명확히 보인다.

확장함수 팁

  • BigDecimal.setScale() 은 객체의 상태를 변경시키지 않음에도 set 이라는 접두사가 붙은 메서드이다.
    • 새로 설정한 정밀도에 맞춘 BigDecimal 을 반환하는 역할이다.
  • 요즘은 수신 객체 상태를 변화시키지 않는 메서드엔 set 이라는 접두사를 붙이지 않는다.
  • 요즘 일반적인 관습으로는 with 접두사를 붙이는 것이 맞다.
amount.withScale(currency.getDefaultFractionDigits())
fun BigDecimal.withScale(int scale, RoundingMode mode) = setScale(scale, mode)

코틀린의 확장 함수를 이용해 역사적인 실수를 바로잡을 수 있다.

코틀린의 Data 클래스

data class EmailAddress private constructor (
    private val localPart: String,
    val domain: String
) {
    companion object {
        fun parse(value: String): EmailAddress {
            val atIndex = value.lastIndexOf('@')
            require(!(atIndex < 1 || atIndex == value.length - 1)) { "EmailAddress must be two parts separated by @" }
            return EmailAddress(
                value.substring(0, atIndex),
                value.substring(atIndex + 1)
            )
        }
    }
}
  • kotlin 의 데이터 클래스는 toString(), equals(), hashCode() 를 대신 구현해준다.
  • 다만 프로퍼티 사이에 불변 조건을 유지해야 하는 값 타입을 데이터 클래스를 사용해 정의하면 안 된다.

위 EmailAddress 생성자의 접근자는 private 이다. parse() 정적 메서드를 통해 클래스를 생성하면 적합한 프로퍼티 값을 가진 클래스만 생성된다. 하지만 data class 를 잘못 사용하여 아래와 같은 일이 일어날 수 있다.

fun main() {
    var email = EmailAddress.parse("me@gmail.com");
    var email2 = email.copy("@@@@", "abc");

    println(email);
    println(email2)
}

data class 는 public copy() 메서드를 생성해내기 때문에 조건을 지키지 않는 새 EmailAddress 값을 만들어낼 수 있다.

또한 값 의미론이 어긋나는 경우에도 data class 를 쓰는 대신 직접 구현하는 편이 좋을 것이다.

소감

  • 자바식으로 코딩할 때 생기는 보일러 플레이트 코드를 많이 줄여주어 우리가 쓸 데 없는 코드를 읽거나 생성하는데 낭비하는 시간을 줄여준다.
  • 다만 data class 는 유의해서 사용하자.
반응형
저작자표시 비영리 (새창열림)

'코틀린 (Kotlin) > 자바에서 코틀린으로' 카테고리의 다른 글

자바에서 코틀린으로 6장 - 자바에서 코틀린 컬렉션으로 요약  (0) 2022.12.06
자바에서 코틀린으로 5장 - 빈에서 값으로 요약  (0) 2022.11.30
자바에서 코틀린으로 4장 - 옵셔널에서 널이 될 수 있는 타입으로 요약  (0) 2022.11.30
    '코틀린 (Kotlin)/자바에서 코틀린으로' 카테고리의 다른 글
    • 자바에서 코틀린으로 6장 - 자바에서 코틀린 컬렉션으로 요약
    • 자바에서 코틀린으로 5장 - 빈에서 값으로 요약
    • 자바에서 코틀린으로 4장 - 옵셔널에서 널이 될 수 있는 타입으로 요약
    Jake Seo
    Jake Seo
    ✔ 잘 보셨다면 광고 한번 클릭해주시면 큰 힘이 됩니다. ✔ 댓글로 틀린 부분을 지적해주시면 기분 나빠하지 않고 수정합니다. ✔ 많은 퇴고를 거친 글이 좋은 글이 된다고 생각합니다. ✔ 간결하고 명료하게 사람들을 이해 시키는 것을 목표로 합니다.

    티스토리툴바