kotlin 学习 (11)

对象表达式与对象声明

有时候,我们需要创建一个对某个类做了轻微改动的类的对象,而不用为之显式声明新的子类。 Kotlin 用对象表达式对象声明处理这种情况。

对象表达式

要创建一个继承自某个(或某些)类型的匿名类的对象,我们会这么写:

window.addMouseListener(object:MouseAdapter() {

overridefunmouseClicked(e:MouseEvent) {/*……*/}

overridefunmouseEntered(e:MouseEvent) {/*……*/}

})

如果超类型有一个构造函数,则必须传递适当的构造函数参数给它。 多个超类型可以由跟在冒号后面的逗号分隔的列表指定:

openclassA(x:Int) {

publicopenvaly:Int=x

}

interfaceB{/*……*/}

valab:A=object:A(1),B{

overridevaly=15

}

任何时候,如果我们只需要“一个对象而已”,并不需要特殊超类型,那么我们可以简单地写:

funfoo() {

valadHoc=object{

varx:Int=0

vary:Int=0

    }

print(adHoc.x+adHoc.y)

}

请注意,匿名对象可以用作只在本地和私有作用域中声明的类型。如果你使用匿名对象作为公有函数的返回类型或者用作公有属性的类型,那么该函数或属性的实际类型会是匿名对象声明的超类型,如果你没有声明任何超类型,就会是 Any。在匿名对象中添加的成员将无法访问。

classC{

// 私有函数,所以其返回类型是匿名对象类型

privatefunfoo()=object{

valx:String="x"

    }

// 公有函数,所以其返回类型是 Any

funpublicFoo()=object{

valx:String="x"

    }

funbar() {

valx1=foo().x// 没问题

valx2=publicFoo().x// 错误:未能解析的引用“x”

    }

}

对象表达式中的代码可以访问来自包含它的作用域的变量。

funcountClicks(window:JComponent) {

varclickCount=0

varenterCount=0

window.addMouseListener(object:MouseAdapter() {

overridefunmouseClicked(e:MouseEvent) {

clickCount++

        }

overridefunmouseEntered(e:MouseEvent) {

enterCount++

        }

    })

// ……

}

对象声明

单例模式在一些场景中很有用, 而 Kotlin(继 Scala 之后)使单例声明变得很容易:

objectDataProviderManager{

funregisterDataProvider(provider:DataProvider) {

// ……

    }

valallDataProviders:Collection<DataProvider>

get()=// ……

}

这称为对象声明。并且它总是在 object 关键字后跟一个名称。 就像变量声明一样,对象声明不是一个表达式,不能用在赋值语句的右边。

对象声明的初始化过程是线程安全的并且在首次访问时进行。

如需引用该对象,我们直接使用其名称即可:

DataProviderManager.registerDataProvider(……)

这些对象可以有超类型:

objectDefaultListener:MouseAdapter() {

overridefunmouseClicked(e:MouseEvent) {……}

overridefunmouseEntered(e:MouseEvent) {……}

}

注意:对象声明不能在局部作用域(即直接嵌套在函数内部),但是它们可以嵌套到其他对象声明或非内部类中。

伴生对象

类内部的对象声明可以用 companion 关键字标记:

classMyClass{

companionobjectFactory{

funcreate():MyClass=MyClass()

    }

}

该伴生对象的成员可通过只使用类名作为限定符来调用:

valinstance=MyClass.create()

可以省略伴生对象的名称,在这种情况下将使用名称 Companion:

classMyClass{

companionobject{ }

}

valx=MyClass.Companion

其自身所用的类的名称(不是另一个名称的限定符)可用作对该类的伴生对象 (无论是否具名)的引用:

classMyClass1{

companionobjectNamed{ }

}

valx=MyClass1

classMyClass2{

companionobject{ }

}

valy=MyClass2

请注意,即使伴生对象的成员看起来像其他语言的静态成员,在运行时他们仍然是真实对象的实例成员,而且,例如还可以实现接口:

interfaceFactory<T>{

funcreate():T

}

classMyClass{

companionobject:Factory<MyClass>{

overridefuncreate():MyClass=MyClass()

    }

}

valf:Factory<MyClass>=MyClass

当然,在 JVM 平台,如果使用 @JvmStatic 注解,你可以将伴生对象的成员生成为真正的静态方法和字段。更详细信息请参见Java 互操作性一节 。

对象表达式和对象声明之间的语义差异

对象表达式和对象声明之间有一个重要的语义差别:

对象表达式是在使用他们的地方立即执行(及初始化)的;

对象声明是在第一次被访问到时延迟初始化的;

伴生对象的初始化是在相应的类被加载(解析)时,与 Java 静态初始化器的语义相匹配。

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 面向对象编程(OOP) 在前面的章节中,我们学习了Kotlin的语言基础知识、类型系统、集合类以及泛型相关的知识。...
    Tenderness4阅读 9,972评论 1 6
  • 原文链接:https://github.com/EasyKotlin 值就是函数,函数就是值。所有函数都消费函数,...
    JackChen1024阅读 11,362评论 1 17
  • val和varval(来自value)——不可变引用。使用val使用val声明的变量不能在初始化之后再次赋值。它对...
    WangSins阅读 3,106评论 0 0
  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 10,232评论 0 5
  • 她不知道要经历多少才能领悟爱的真谛,一直以来他离开以后她就一个人走走停停,难过了就自己和着发烫的洗澡水狠狠的哭,开...
    温仪阅读 3,262评论 0 1

友情链接更多精彩内容