Kotlin学习笔记(二)

构造函数

在Kotlin中一个类可以有一个主构造函数和一个或多个的次构造函数,主构造函数是类头的一部分

如果一个非抽象类没有声明任何(主或次)构造函数,它会有一个生成的 不带参数的主构造函数,构造函数的可见性是 public

主构造函数

/**
 *  主构造函数
 */
class Kotlin constructor(key: String, value: String) {

}

/**
 *  省略了constructor关键字的主构造函数
 *  NOTE:这个主构造函数不能包含任何代码, 初始化代码可以放到init关键字的代码块
 */
class Computer(price: Int) {
    var price: Int = 0
    init {
        this.price = price
    }
}

/**
 * 如果构造函数有注解修饰或者有可见性(public, private...)修饰符修饰
 * 则必须有关键字constructor
 */
class Book public constructor(name: String) {
    
}

/**
 * 没有类体的类,可以省略大括号
 * NOTE: 同时声明了mobile为只读属性,brand参数为可变参数
 */
class Phone(val mobile: String, var brand: String)

/**
 * 将构造函数私有
 */
class PrivateClass private constructor() {
    
}

次构造函数

  • 次构造函数必须声明前缀constructor
  • NOTE: 如果一个类有主构造函数,则次构造函数必须委托给主构造函数, 可以直接委托或者通过别的次构造函数间接委托
/**
 * 有主构造函数,也有次构造函数的类
 */
class Glass(val name: String) {
    var mName: String = ""
    var mBand: String = ""
    init {
        this.mName = name
    }
    constructor(name: String, brand: String) : this(name) {
        this.mName = name
        this.mBand = brand
    }
}

创建类实例

  • Kotlin没有new关键字
var phone: Phone = Phone("Kotlin")

继承

  • Kotlin中所有类都有一个超类Any(类似于Java的Object, 但不等同于Object)
  • Kotlin中所有类默认都是不可以继承的(即所有类都是final), 如果想让一个类可以被继承,可以使用关键字open
  • 如果父类有主构造函数,子类必须使用父类的朱构造函数就地初始化
  • 如果父类没有主构造函数,但是有次构造函数,则子类的每个次构造函数必须使用super关键字初始化其类型,或者委托给另一个构造函数
/**
 * 指明该类可以被继承
 */
open class Human(name: String) {

    var mName: String = ""

    init {
        this.mName = name
    }

    /**
     * 显示的标注该方法可以被覆盖重写
     */
    open fun method() {

    }

    /**
     * 该方法不可以被重写
     */
    fun method1() {

    }
}

/**
 * 用父类的主构造函数就地初始化
 */
class Man(name: String) : Human(name) {
    /**
     * 重写父类的method方法
     */
    override fun method() {
        super.method()
    }
}

open class Woman(name: String) : Human(name) {
    /**
     * 禁止Woman的子类覆盖重写该方法
     */
    final override fun method() {
        super.method()
    }
}


/**
 * super关键字
 */
class MyView : View {

    constructor(context: Context) : super(context) {

    }

    constructor(context: Context, attrs: AttributeSet) : super(context, attrs) {

    }


}

  • 如果一个类从它的直接父类继承相同成员的多个实现,它必须覆盖这个成员,并提供自己的实现,可以通过super<?>的方式指定采用哪个父类的实现

open class A {
    open fun foo() {
        // ...
    }
}

interface B {
    /**
     * 有方法体
     */
    fun foo() {
        // ...
    }
}

/**
 * A没有提供构造函数,所以有默认的空的主构造函数
 * 这里必须使用空构造函数来就地初始化
 */
class C : A(), B {
    override fun foo() {
        super<A>.foo()
        /**
         * 这里如果要调用interface的foo, B.foo必须有方法体(即实现了该方法)
         * 如果只是简单的声明方法(和Java中interface的用法一样),则这里无法直接调用
         * super<B>.foo()
         */
        super<B>.foo()
    }
}

  • 抽象类可以不用open关键字修饰

open class Base {
    open fun foo() {}
}

abstract class Child : Base() {
    override abstract fun foo()
}

密封类

  • 类似于Enum类,是Enum类功能的扩展
  • 密封类使用关键字sealed修饰
  • 密封类的所有子类必须嵌套声明在该密封类的内部, 不过子类的子类不用必须声明在密封类内部,可以在任何地方声明

使用密封类的关键好处在于使用 when 表达式 的时候,如果能够 验证语句覆盖了所有情况,就不需要为该语句再添加一个 else 子句了

sealed class Expr {
    class Const(val number: Double) : Expr()
    class Sum(val e1: Expr, val e2: Expr) : Expr()
    /* 对象声明 */
    object NotANumber: Expr()

}

fun testExpr(expr: Expr): Double = when (expr) {
    is Expr.Const -> expr.number
    is Expr.Sum -> testExpr(expr.e1) + testExpr(expr.e2)
    Expr.NotANumber -> Double.NaN

}

声明属性

  • val代表只读属性,不允许setter
  • var代表可变属性
  • 声明一个属性的完整语法是:
var <propertyName>: <propertyType> [= <property_initializer>]
 [<getter>]
 [<setter>]

class Test(size: Int) {

    /**
     * 定义getter和setter
     */
    var mSize: Int?
        get() = this.mSize
        set(value) {
            this.mSize = value
        }

    init {
        this.mSize = size
    }

    /**
     * 只读变量只有getter
     */
    val isEmpty: Boolean
        get() = this.mSize == 0

    /**
     * 改变setter的可见性,但不改变默认的视线
     */

    var setterVisibility: String = "abc"
        private set

    /**
     * 可以使用field标识符来访问属性,field标识符只能用在属性的访问器内
     */
    var counter = 0
        set(value) {
            if (value >= 0)
                field = value
        }

}

编译期常量

编译期常量,可以用const关键字标识某个属性为编译期常量,一个编译期常量必须具有以下要求:

  1. 在top-level声明(直接在包内声明)或是一个object的成员
  2. String或者基础类型,而且有初始值
  3. 没有自定义的getter方法
const val DEPRECIATED: String = "depreciated"

class Const {
    @Deprecated(DEPRECIATED) fun foo(){ }
}

延迟初始化

在Kotlin中,被定义为非空类型的属性,都需要初始化,如果想之后通过某个方法再对该属性进行初始化,可以使用lateinit修饰符

class LateInit {
    lateinit var mName: String

    fun setup(name: String) {
        mName = name
    }

    /**
     * 通过依赖注入的方式初始化
     */
    lateinit var mChild: Child
    
    fun setChild(child: Child) {
        this.mChild = child
    }
}

NOTE: 对于lateinit有以下几点限制:

  1. 只能用在var类型的可变属性
  2. 被修饰的属性不可以有自定义的gettersetter
  3. 属性必须是非空的
  4. 不可以用在基本类型上

接口

  • Kotlin中的接口即可以有方法的声明,也可以包含方法的实现
  • 接口的属性要么是抽象的,要么提供getter
interface MyInterface {

    val property: Int //抽象属性
    
    val test: String //提供访问器的属性
        get() = "test"

    /**
     * 只是声明
     */
    fun foo()

    /**
     * 有实现
     */
    fun bar() {

    }
}

class Impl: MyInterface {

    override val property: Int
        get() = 10 
        
    override fun foo() {
        val str = test
        Log.d("tag", str)
    }

}

Override冲突

如果实现多个接口,接口方法名同名时,可以用super<?>来解决


interface A1 {
    fun foo() {
        
    }
}

interface B1 {
    fun foo() {
        
    }
}


class C1: A1, B1 {
    override fun foo() {
        super<A1>.foo()
        super<B1>.foo()
    }
    
}

可见性

Kotlin有四种可见性修饰符:

  • public
  • private
  • protected
  • internal

默认的修饰符是public

局部变量不允许有可见性修饰符

顶层声明

直接在包内声明叫做顶层声明,函数,属性,类,对象,接口都可以在顶层声明

protected不适用于顶层声明

package test

//只在当前文件内可见
private fun test() {}

//随处可见
public var str: String = "str"

//test随处可见, 但是test的setter方法只在当前kt文件内可见
public var test: String = "test"
    private set 

//对同一模块内的文件都可见
internal var test1: Int = 1

类和接口中可见性

在类和接口中,四个可见性修饰符的意义是:

  • public 随处可见
  • private 只在这个类内部可见
  • protected 只在这个类内部以及其子类中可见
  • internal 对于同属于一个模块的都可见

模块的定义

模块在Kotlin中就是一系列的Kotlin文件编译在一起

  • 一个IntelliJ IDEA module
  • a Maven or gradle project
  • a set of files compiled with one invocation of the Ant task

扩展函数

如果要给一个已经定义好的类新增一个函数,可以使用扩展函数功能

扩展函数并没有给类新增成员,仅仅是通过该类的实例去调用这个新函数

/**
 * 给MutableList扩展一个swpa函数
 */
fun <T> MutableList<T>.swap(i1: Int, i2: Int) {
    val tmp = this[i1]
    this[i1] = this[i2]
    this[i2] = tmp
}

/**
 * 调用扩展的swap函数
 */
fun test1() {
    val list = mutableListOf<Int>(1, 2, 3)
    list.swap(0, 2)
}

属性扩展

扩展属性只能通过明确的settergetter进行定义,不可以直接进行初始化

class Test1 

class Test2 {
    var Test1.name: String
        get() = "kotlin"
        set(value)  {
            name = value
        }
    
    fun test2() {
        var test1 = Test1()
        test1.name = "haha"
    }
}

嵌套类

在类的内部可以嵌套其他的类:

class Out {
    private val outStr: String = "out"
    //并不能直接访问Out的属性和方法
    class In {
        fun foo() = 2
    }

    /**
     * 内部类会带有外部类的引用, 可以访问外部类的属性和方法
     */
    inner class Inner {
        fun foo() = outStr
    }
}

内部类用inner标记,内部类会带有一个来自外部类的对象的引用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,265评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,078评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,852评论 0 347
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,408评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,445评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,772评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,921评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,688评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,130评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,467评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,617评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,276评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,882评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,740评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,967评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,315评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,486评论 2 348

推荐阅读更多精彩内容