일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
- OnlineIDE
- docker
- linux job
- 네이버클라우드
- git
- 리눅스 파일동기화
- Notification
- 3D모델링
- 3D
- Token인증방식
- 코틀린개발환경
- data serialization formats
- NAS HDD교체
- C언어
- octoprint
- IoTNetwork
- 데이터 직렬화 포맷
- gitlab
- ci/cd
- 3D프린터입문
- kubernetes
- 3D 프린터 용어
- github
- nas
- Kotlin
- docker container
- 3D프린터 입문
- 3d프린터 초보
- Jenkins
- 3D프린터
- Today
- Total
0과 1을 공부하다.
[Kotlin] 코틀린 클래스 본문
접근 제어자
자바의 접근제어 (가시성 제어)
- public : 모든 곳에서 접근 가능
- protected: 같은 패키지 또는 하위 클래스에서만 접근 가능
- default: 같은 패키지에서만 접근 가능
- private: 선언된 클래스 내에서만 접근 가능
코틀린의 접근 제어 (가시성 제어)
- public : 모든 곳에서 접근 가능
- protected: 선언된 클래스 또는 하위 클래스에서만 접근 가능
- internal: 같은 모듈에서만 접근 가능 (한 번에 컴파일 되는 코드)
- private: 선언된 클래스 내에서만 접근 가능
코틀린 파일 접근 제어
- public : 기본값 어디서든 접근할 수 있다.
- protected: 파일 최상단에서는 사용할 수 없다.
- internal: 같은 모듈에서만 접근 가능
- private: 같은 파일 내에서만 접근 가능
클래스의 생성자에 접근 지시자를 붙이려면 constructor을 붙어야 한다.
class Person private constructor () {
}
클래스 선언 및 인스턴스 생성
- 기본 선언 및 인스턴스 생성
fun main() {
val jay = Person("jay", 30)
println("name: " + jay.name)
println("age: " + jay.age)
}
// 생성자를 통해 멤버 변수를 선언하고 초기화 할 필요 없음
class Person (
var name: String, // public / private / protected / internal 를 통해 접근제한을 할 수 있다.
var age: Int,
)
- 생성자 혹은 프로퍼티 정의
: init{} 생성자가 생성되는 시점에 1회 실행되는 블록
: constructorsms{} 생성자 재정의 -> 주 생성자는 반드시 존재해야 한다. → 최종적으로 주생성자를 호출해야 한다. → 바디를 가질 수 있다.
class Person (
private var name: String,
public var age: Int,
) {
var hobby = "" // 생성자 내부 이외의 별도의 프로퍼티
init { // 초기화 블록
print("init")
}
}
- data class
: 데이터 보관용 객체를 만들 때 사용하는 특별한 종류의 클래스
fun main() {
val jay = Person("jay", 30)
val jay2 = Person("jay", 30)
println(jay == jay2) // calss 정의에서 data 키워드를 통해 객체 내부의 값을 비교하여 판별함.
}
data class Person (
private var name: String,
public var age: Int,
)
- Singleton class : 단 하나의 인스턴스만을 갖는 클래스
object Singleton {
var counter: Int = 0
fun printCounter() {
println("Counter: $counter")
}
fun incrementCounter() {
counter++
}
}
fun main() {
Singleton.printCounter() // Counter: 0
Singleton.incrementCounter()
Singleton.printCounter() // Counter: 1
}
- Enum class
: eum의 특징은 추가적인 클래스를 상속받을 수 없으며 인터페이스는 구현할 수 있으며, 각 코드가 싱글톤이다.
enum class Country (
private val code: String,
){
KOREA("KO"),
AMERICA("US")
;
}
- getter setter
: 프로퍼티 = 필드 + getter + setter를 의미한다. 기존 자바에서는 getter와 setter를 정의해야 하지만 코틀린은 기본적으로 프로퍼티에 대해 자동으로 getter와 setter를 생성해 주지만, 필요에 따라 사용자 정의 getter와 setter를 만들 수 있다.
fun main() {
val jay = Person("jay", 30)
println(jay.hobby)
jay.hobby = "baseball"
println(jay.hobby)
}
class Person (
private var name: String,
public var age: Int,
) {
var hobby = "football"
// private set // set private
set(value) {
// field 키워드를 사용하여 backing field에 접근
field = value.uppercase() // 커스텀(사용자 정의) setter
}
get() = field // 커스텀(사용자 정의) getter
}
static 함수와 변수
: 코틀린에서는 static은 존재하지 않는다. 대신 companion object 블럭을 사용하여 구현한다.
class Calculator(){
companion object {
const val PI: Double = 3.14159 // const 키워드를 붙으면 런타임시에 할당되는 것이 아닌 컴파일시에 값이 할당됨.
fun sum(x: Int, y: Int): Int = x + y
fun circumference(radius: Int): Double = 2 * PI * radius
}
}
fun main(){
println(Calculator.PI)
println(Calculator.sum(1,2))
println(Calculator.circumference(5))
}
상속(Extends)
상속(Extends)을 설명하기 이전에 abstract 키워드와 open 키워드에 대해 설명한다.
- abstract:
- 클래스: 인스턴스화할 수 없고, 반드시 다른 클래스에 의해 상속되어야 한다.
- 메서드: 구현이 없고, 반드시 하위 클래스에서 구현해야 한다.
- open:
- 클래스: 기본적으로 상속할 수 없는 클래스에 대해 상속 가능하게 한다.
- 메서드: 기본적으로 재정의할 수 없는 메서드에 대해 재정의 가능하게 한다.
- 추상 프로퍼티가 아니라면 상속받을 때 꼭 open을 붙인다.
- 프로퍼티를 오버라이딩 할 때는 선언에서 open 키워드를 붙어야 한다.
- final: overrride를 할 수 없게 한다.
fun main() {
val dog = Dog()
val cat = Cat()
println(dog.move())
println(cat.move())
}
// 기본적으로 일반 클래스는 상속이 불가능하며, open 키워드를 붙어야 함. abstract = X
abstract class Animal {
open fun move() { // 'open'은 override 가능 상태로 만든다.
println("Animal move")
}
}
class Dog: Animal() // 반드시 생성자 호출 해야함 {
// Override
override fun move() {
println("Dog move")
}
}
class Cat: Animal() {
// Override
override fun move() {
println("Cat move")
}
}
추가로 상위 클래스를 설계할 때 생성자 또는 초기화 블록에 사용되는 프로퍼티에는 open을 피해야 한다.
인터페이스(Interface)
fun main() {
val dog = Dog()
val cat = Cat()
println(dog.move())
println(cat.move())
}
interface Drawable {
fun draw()
}
abstract class Animal {
open fun move() { // 'open'은 override 가능 상태로
println("Animal move")
}
}
class Dog: Animal(), Drawable {
// Override
override fun move() {
println("Dog move")
}
override fun draw(){
println("Dog draw")
}
}
class Cat: Animal(), Drawable {
// Override
override fun move() {
println("Cat move")
}
override fun draw(){
println("Dog draw")
}
}
객체 타입체크 is / 강제 타입 변환 as
// 상위 클래스 정의
open class Animal {
open fun move() {
println("Animal move")
}
}
// Dog와 Cat 클래스는 Animal을 상속받음
class Dog : Animal() {
override fun move() {
println("Dog move")
}
}
class Cat : Animal() {
override fun move() {
println("Cat move")
}
}
fun main() {
val dog: Animal = Dog()
val cat: Animal = Cat()
// 생성된 인스턴스의 클래스 비교
if (dog is Cat) {
println(dog.move())
} else {
println("dog is not a Cat")
}
// 타입 변환
val test = cat as? Dog
println(test) // Output: null
}
제네릭(Generic)
: 타입 안전성을 높이고 코드의 재사용성을 증가시키기 위해 사용되는 기능이다. 클래스, 함수, 인터페이스 등을 정의할 때 타입을 매개변수화하여 다양한 타입의 객체들을 다룰 수 있다.
- 제네릭 클래스
fun main() {
val boxInt = Box(10)
val boxString = Box("BOX")
println(boxInt.value)
println(boxString.value)
}
class Box<T>(var value: T)
- 제네릭 함수
fun <T>Box(value: T): T
{
return value
}
fun main() {
val boxInt: Int = Box(10)
val boxString: String = Box("BOX")
println(boxInt)
println(boxString)
}
구조분해
: 복합적인 값을 분해하여 여러 변수를 한 번에 초기화 하는 것
data class Person(val name: String, val age: Int)
fun main() {
val person = Person("박지환", 29)
val (name, age) = person // 프로퍼티 순서에 따라 할당됨.
println("이름: ${name} | 나이: ${age}")
}
확장 함수 / 중위 함수
확장 함수
: 기존 클래스에 새로운 함수를 추가할 수 있는 기능
- 확장함수는 클래스에 있는 private 또는 protected 멤버를 가져올 수 없다.
- 확장함수와 멤버함수의 시그니처가 같으면 멤버함수가 우선적으로 호출된다.
- 확장함수는 현재 타입을 기준으로 호출된다. 즉, 오버라이딩 되어도 타입을 기준으로 호출한다.
// String 클래스에 확장 함수 추가
fun String.addExclamation(): String {
return "$this!"
}
fun main() {
val hello = "Hello"
val helloWithExclamation = hello.addExclamation()
println(helloWithExclamation) // "Hello!" 출력
}
중위 함수(infix)
: .과 ()를 생략하고, 연산자처럼 사용할 수 있는 함수
class Calculator(val x: Int){
infix fun sum(y: Int) : Int = x + y
}
fun main(){
val obj = Calculator(1)
println(obj sum 2)
}
※ 본 게시글의 정보가 잘못 되었거나 부족한 부분에 대한 피드백을 환영합니다.
* CopyRight 2024. Jay Park All rights reserved.
'Study > Kotlin' 카테고리의 다른 글
[Kotlin] 코틀린 참고하기 (2) | 2024.07.02 |
---|---|
[Kotlin] 코틀린 코루틴 (247) | 2024.07.02 |
[Kotlin] 코틀린 함수 (2) | 2024.07.02 |
[Kotlin] 코틀린 조건문 / 반복문 (2955) | 2024.07.02 |
[Kotlin] 코틀린 리스트/배열 (64) | 2024.07.02 |