Kotlin中那些特别的

类(Class)是面向对象程序设计(OOP,Object0Oriented Programming)实现信息封装的基础。包含属性和方法……

以上是摘抄自百度百科的关于描述,作为Android开发者,我们接触最多的无过于Java以及现在火热的Kotlin

不同于Java中相对中规中矩的通用简一的类定义方式,在Kotlin中有了较多的关键字类定义一些特别的类,比如单例类伴生内部类密封类数据类等,对比于Java我们来分析一下这些特别的类,会不会让你学的特别累

一、简化的数据类

数据类(data class),用于保存元数据的封装类,Java中的POJO(Plain Ordinary Java Object)所有都是继承自Object,并自然而然的有其toString()hashcode()equals()等函数。一般都需要有getter/setter,复杂的Java Bean的话,手写getter/setter实在是挺繁琐的,即使有些快捷框架,也未必能尽如人意。

public class Student {
    private String name;
    private int age;
    private String desc;

    public Student() {
        //无参构造函数
    }

    /**
     * 多参数构造函数
     * @param name
     * @param age
     * @param desc
     */
    public Student(String name, int age, String desc) {
        this.name = name;
        this.age = age;
        this.desc = desc;
    }

    //getter/setter
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    //...其他setter getter

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", desc='" + desc + '\'' +
                '}';
    }
}

Kotlin中对应数据类data class

data class Student(val name:String,var age:Int,var desc:String)

查看kotlin转java的代码,就会感觉的kt的甜甜蜜蜜

在AS中Tools--Kotlin--ShowKotlinBytecode得到Kotlin字节码,点击Decompile得到对应的Java代码形式

android kotlin内部类使用 kotlin object类_Java

数据类的知识点:

  1. 和其他Kotlin的定义类一样,都是默认final的;
  2. 可以看出Kotlinnullable可空类型是区别的,StringString?不一样。图中可见,反编译后对应Java写法就是@NotNull的注解;
  3. data class是专用的数据封装类,它的构造函数至少有一个主构造函数及至少一个入参,主构造函数的入参必须有val/var的修饰(这么才算做是类的属性而非简单的构造入参);
  4. 数据类会将构造函数内的参数依顺序生成component123之类的,便于读取,自然也根据权限修饰符来确定属性的getter/setterval的是没有setter的;
  5. 可通过默认值的形式来实现其他无参构造的需求data class AA(val name:String="")
  6. 类中声明的属性,如果data class不复写toString()equals()hashcode(),那么默认判定对象特征值只是用到构造函数内的属性参数,也就会出现即使类内属性值不一致而认定为相等,记住是相等而不是同一对象
//name是构造属性
data class People(val name:String){
    var age:Int = 18//类内属性
}
//构造参数只有name,如果一致,即使age不一致
val p1 = People("张三")
val p2 = People("张三")
p1.age = 20
p2.age = 30
//若没有复写equals、hashcode,就会会判定为两个对象相等,
p1== p2 就是true
//切记!! 相等的原因是没有override equals和hashcode,但是对象本身并不是同一个,
p1===p2 是false

数据类有copy函数,不同于JavacloneKotlin中的copy就是便捷的帮你new了对象并赋值了原有对象的参数(根据你的修改与否)

@Test
    fun testKt(){

        val a1 = AA("张三")
        val a2 = AA("张三")
        val a1Copy  = a1.copy()
        val a2Copy  = a2.copy(name = "李四")
        a1.age = 19
        a2.age = 29
        println("a1===a2 ${a1 === a2}")
        println("a1==a2 ${a1 == a2}")
        println("a1==aCopy ${a1 == a1Copy}")
        println("a1===a1Copy ${a1 === a1Copy}")
        println("a1==a2Copy ${a2 == a2Copy}")
        println("a1===a2Copy ${a2 === a2Copy}")

    }
//输出结果
a1===a2 false
a1==a2 true
a1==aCopy true
a1===a1Copy false
a1==a2Copy false
a1===a2Copy false

看一下对应java代码,可以看出copy就是甜甜的语法糖,帮你new

android kotlin内部类使用 kotlin object类_ide_02

  1. 数据类的解构声明
    可能不好理解这个词,看代码就清晰了
data class AA(val name:String,var age:Int,var desc:String)
//定义一个函数,返回AA对象
fun getAA():AA{
    return AA("zhangsan",39,"a worker")
}
//调用处,就可感觉到解构声明的魅力
val aaa = getAA()//普通的方式
val (name,age,desc) = getAAA()//解构声明的方式,便于直接使用某些参数,而不需要aaa.name
  1. 系统标准库提供了PairTriple数据类,分别是两个参数和三个参数的。
  2. 封装性的数据类,可使用泛型方式定义参数类型
//这样数据类可以使用泛型确定内部参数类型,也有其特定使用场景
data class BBB<T,R,Q,S>(val t:T,var r:R,var q:Q,val s:S)
二、不是🐝甜似🍯的密封类

密封类用以表示受限的类继承结构

快速理解,类似于大号的枚举,用于特定类型限定。与枚举异同

  • 枚举,一种特定类型,可有多个枚举常量。每个枚举值只是一个实例。
  • 密封类,可有很多子类,每个子类都可有多个实例。
  1. 密封类的使用特性注意点
  • sealed关键字,其所有子类都必须在同一kt文件内,且最好是top level的;
  • 若在其他类内声明,则其子类就只能在自身内部声明了;
  • 密封类是抽象的,可有抽象成员,但不能实例化;
  • 密封类构造函数私有;
  1. 演示更直白
class ExampleUnitTest {
    //在其它类内声明,则其子类也就只能在其内部了,根源在于sealed class的私有化构造函数
    sealed class HHH() {
        object mmm : HHH()
    }

    object hhhhhhh : HHH()//无法在其它类内继承 密封类

    data class eeeeee(val name: String) : AAAA() {
        class eee(val age: Int) : AAAA()
    }

}

//在外也不能继承 其他类内 看似不报错的密封类
object ggg : ExampleUnitTest.HHH()


//这才是合规的
sealed class AAAA {
    fun aa() {}
    val bbb: String = ""
    open fun ad() {}
    abstract class ccc() {}

    //在它自身内部可以
    object jjj : AAAA()
}

data class ccc(val name: String) : AAAA() {
}

class ddd(val age: Int) : AAAA()
object fff : AAAA()

android kotlin内部类使用 kotlin object类_ide_03


android kotlin内部类使用 kotlin object类_Java_04

三、Object

Object类可以快速创建Kotlin版本的单例类,也可以是companion object的伴生类,其特性差不多。

object单例类,也是kotlin的类的一种,比较特殊而已。

  1. 私有化构造函数,且无参数;也就是说,不能显式出构造函数,也不能有次级构造函数;
  2. 不能open/abstract,内部也不能有open的函数
  3. 其反编译为Java代码,也就是静态饿汉式的单例类写法,线程安全的。
四、枚举类

Kotlin中的枚举类,类比于Java的枚举,更为灵活一些

enum class Direction{
    NORTH,SOUTH,WEST,EAST//枚举对象用,符号分隔
}
enum class Color(val colorName:String){//这里添加val/var为的是可以称为成员属性参数
    YELLOW("#f0f0f0"), GREEN("#00f0f0"), BLUE("#000ff0");
}
  1. 类似于Java枚举,Kotlin的枚举可以有参数构造函数
  2. 不同于Java枚举,Kotlin枚举可以有抽象函数,这样每个实例都要override改函数
enum class ColorHHH(val cn: String) {
    //每个实例,都要override
    YELLOW("#f0f0f0"){
        override fun info() {
            TODO("Not yet implemented")
        }
    }, GREEN("#00f0f0"){
        override fun info() {
            TODO("Not yet implemented")
        }
    }, BLUE("#000ff0"){
        override fun info() {
            TODO("Not yet implemented")
        }
    };
    //定义枚举类的抽象函数,其内部实例,就要override
    abstract fun info()
    open fun foo(){}
    fun boo(){}
}
  1. EnumClass.valueOf(value:String)这个函数,入参是String就是对应枚举实例对象的名字。
val co = ColorHHH.valueOf("YELLOW")//这样才能获取到
co.name//就是实例的名字,co.ordinal,就是实例在枚举类中定义的索引编号。
  1. 枚举类可以实现接口,但是不能继承类。当然也不能open/abstract。同2,如果实现接口,所以实例都要实现,或者枚举类通用实现。
interface Color9{
    fun fillColor()
}
interface Shape{
    fun defineShape()
}
//实现上面两个接口
enum class Car:Color9,Shape{
    //1、每个实例都实现接口函数的方式
    BIZ{
        override fun fillColor() {
            TODO("Not yet implemented")
        }

    },BW{
        override fun fillColor() {
            TODO("Not yet implemented")
        }
    };
	//2,或者就是枚举类自身直接实现接口的函数
    override fun defineShape() {
        
    }
}
五、内部类、嵌套类
  • 嵌套类,顾名思义,嵌套在其它类中的定义类。Kotlin的特别之处,接口和类可以互相嵌套,也就是类中可定义接口,接口中可定义类。
interface OuterInterface {
    class InnerClass
    interface InnerInterface
}

class OuterClass {
    class InnerClass
    interface InnerInterface
    inner class RealInnerClass//这才是对应与Java的内部类,会引用外部类的对象
}
//匿名内部类,如果java的new XXX直接用一样
window.addMouseListener(object:MouseAdapter(){
    override fun mouseClicked(e: MouseEvent){
        //...
    }
    override fun mouseEntered(e: MouseEvent){
        //...
    }
})

注意,kotlin中这么写是嵌套类,在Java中这么写就是内部类了。

  • 内部类,不同于Java中的写法,这里需要一个标记inner
  • 匿名内部类,Kotlinobject的另一作用