Kotlin的对象表达式与Java中的匿名内部类的主要区别:匿名内部类只能指定一个父类型,但对象表达式可以指定0~N个肤类型。

一、对象表达式

对象表达式的语法格式如下:

object [: 0~N个父类型]{

//对象表达式的类体部分

}

对象表达式还有如下规则:

对象表达式不能是抽象类,因为系统在创建对象表达式时会立即创建对象。因此不允许将对象表达式定义成抽象类。

对象表达式不能定义构造器。但对象表达式可以定义初始化块,可以通过初始化块来完成构造器需要完成的事情。

对象表达式可以包含内部类,不能包含嵌套类。

package `0705`

interface Outputable {

fun output(msg: String)

}

abstract class Product(var price: Double) {

abstract val name: String

abstract fun printInfo()

}

fun main(args: Array) {

//指定一个父类型(接口)的对象表达式

var ob1 = object : Outputable {

override fun output(msg: String) {

for (i in 1..6) {

println("${msg}")

}

}

}

ob1.output("随便输出点什么吧")

println("-----------------------------------------------")

//指定零个父类型的对象表达式

var ob2 = object {

//初始化块

init {

println("初始化块")

}

//属性

var name = "Kotlin"

//方法

fun test() {

println("test方法")

}

//只能包含内部类,不可以包含嵌套类

inner class Inner

}

println(ob2.name)

ob2.test()

println("-----------------------------------------------")

//指定两个父类型的对象表达式

var ob3 = object : Outputable, Product(1.23) {

override fun output(msg: String) {

println("输出信息:${msg}")

}

override val name: String

get() = "激光打印机"

override fun printInfo() {

println("高速极光打印机们支持自动双面打印!")

}

}

println(ob3.name)

ob3.output("Kotlin慢慢学")

ob3.printInfo()

}

输出结果:

随便输出点什么吧

随便输出点什么吧

随便输出点什么吧

随便输出点什么吧

随便输出点什么吧
随便输出点什么吧

-----------------------------------------------

初始化块

Kotlin

test方法

-----------------------------------------------

激光打印机

输出信息:Kotlin慢慢学

高速极光打印机们支持自动双面打印!

Kotlin的对象表达式可分为两种情形:

对象表达式在方法的局部范围内,或使用private修饰的对象表达式,Kotlin编译器可识别对象表达式的真实类型。

非private修饰的对象表达式与Java的匿名内部类相似,编译器只会把对象表达式当成它所继承的父类或所实现的接口处理。如果它没有父类型,系统当它是Any类型。

package `0705`

class ObjectExprType {

private val ob1 = object {

val name: String = "Kotlin"

}

internal val ob2 = object {

val name: String = "Kotlin"

}

private fun privateBar()=object {

val name:String="Java"

}

fun publicBar()=object {

val name:String="Java"

}

fun test(){

//ob1是private对象表达式,编译器可识别它的真实类型

println(ob1.name)

//ob2是非private对象表达式,编译器当它是Any类型

// println(ob2.name)

//privateBar是private函数,编译器可识别它返回的对象表达式的真实类型

println(privateBar().name)

//publicBar是非private函数,编译器将它返回的对象表达式当成Any类型

// println(publicBar().name)

}

}

fun main(args: Array) {

ObjectExprType().test()

}

输出结果:

Kotlin

Java

Kotlin编译器可以识别private对象表达式的真实类型。

Kotlin的对象表达式可访问或修饰其作用域内的局部变量。

fun main(args: Array) {

var a = 20

var obj = object {

fun change() {

println("change()方法修改变量a的值")

a++

}

}

obj.change()

println(a)

}

输出结果:

change()方法修改变量a的值

21

Kotlin的对象表达式比Java的匿名内部类增强了三个方面:

对象表达式可指定多个父类型

Kotlin编译器能更准确地识别局部范围内private对象表达式的类型。

对象表达式可访问或修改其所在范围内的局部变量

二、对象声明和单例模式

对象声明的语法格式如下:

object ObjectName [: 0~N个父类型]{

//对象表达式的类体部分

}

对象声明与对象表达式的语法很相似,区别在于:对象表达式在object关键字后没有名字;而对象声明需要在object关键字后指定名字。

两者还有如下区别:

对象表达式是一个表达式,可以被赋值给变量;而对象声明不是表达式,不能用于赋值。

对象声明可包含嵌套类,不能包含内部类;而对象表达式可包含内部类,不能包含嵌套类。

对象声明不能定义在函数和方法内;但对象表达式可嵌套在其他对象声明或非内部类中。

package `0705`

interface Outputable {

fun output(msg: String)

}

abstract class Product(var price: Double) {

abstract val name: String

abstract fun printInfo()

}

//指定一个父类型的对象表达式

object MyObject1 : Outputable {

override fun output(msg: String) {

for (i in 1..6) {

println("${msg}")

}

}

}

//指定零个父类型的对象表达式

object MyObject2 {

//初始化块

init {

println("初始化块")

}

//属性

var name = "Kotlin"

//方法

fun test() {

println("test方法")

}

//只能包含嵌套类,不可以包含内部类

class Inner

}

//指定两个父类型的对象表达式

object MyObject3 : Outputable, Product(1.23) {

override fun output(msg: String) {

println("输出信息:${msg}")

}

override val name: String

get() = "激光打印机"

override fun printInfo() {

println("高速极光打印机们支持自动双面打印!")

}

}

fun main(args: Array) {

MyObject1.output("一起来学Kotlin")

println("-----------------------------------------------")

println(MyObject2.name)

MyObject2.test()

println("-----------------------------------------------")

println(MyObject3.name)

MyObject3.output("Kotlin真不错")

MyObject3.printInfo()

}

输出结果:

一起来学Kotlin

一起来学Kotlin

一起来学Kotlin

一起来学Kotlin

一起来学Kotlin
一起来学Kotlin

-----------------------------------------------

初始化块

Kotlin

test方法

-----------------------------------------------

激光打印机

输出信息:Kotlin真不错

高速极光打印机们支持自动双面打印!

对象声明专门用于实现单例模式,对象声明所定义的对象也就是该类的唯一实例,程序可通过对象声明的名称直接访问该类的唯一实例。

三、伴生对象和静态成员

在类中定义的对象声明,可使用companion修饰,这样该对象就变成了伴生对象。

每个类最多只能定义一个伴生对象,伴生对象相当于外部类的对象,程序可通过外部类直接调用伴生对象的成员。

package `0705`

interface CompanionTest {

fun output(msg: String)

}

class MyClass {

//使用companion修饰的伴生对象

companion object MyObject1 : CompanionTest {

val name = "name属性值"

override fun output(msg: String) {

for (i in 1..6) {

println("${msg}")

}

}

}

}

fun main(args: Array) {

//使用伴生对象所在的类调用伴生对象的方法

MyClass.output("Kotlin必须学")

println(MyClass.name)

}

输出结果:

Kotlin必须学

Kotlin必须学

Kotlin必须学

Kotlin必须学

Kotlin必须学
Kotlin必须学

name属性值

伴生对象的主要作用就是为其所在的外部类模拟静态成员,但只是模拟,伴生对象的成员依然是伴生对象本身的实例成员,并不属于伴生对象所在的外部类。

四、伴生对象的扩展

伴生对象也可以被扩展。如果一个类具有伴生对象,则Kotlin允许为伴生对象扩展方法和属性。

package `0705`

interface CompanionTest {

fun output(msg: String)

}

class MyClass {

//使用companion修饰的伴生对象

companion object : CompanionTest {

val name = "name属性值"

override fun output(msg: String) {

for (i in 1..6) {

println("${msg}")

}

}

}

}

//为伴生对象扩展方法

fun MyClass.Companion.test() {

println("为伴生对象扩展的方法")

}

val MyClass.Companion.foo

get() = "为伴生对象扩展的属性"

fun main(args: Array) {

//使用伴生对象所在的类调用伴生对象的方法

MyClass.output("Kotlin必须学")

println(MyClass.name)

//通过伴生对象所在的类调用为伴生对象扩展的成员

MyClass.test()

println(MyClass.foo)

}

输出结果:

Kotlin必须学

Kotlin必须学

Kotlin必须学

Kotlin必须学

Kotlin必须学
Kotlin必须学

name属性值

为伴生对象扩展的方法

为伴生对象扩展的属性

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

android对象申明,Kotlin中的对象表达式和对象声明的具体使用相关推荐

  1. 【Kotlin】Kotlin 中使用 Lambda 表达式替代对象表达式原理分析 ( 尾随 Lambda - Trailing Lambda 语法 | 接口对象表达式 = 接口#函数类型对象 )

    文章目录 一.尾随 Lambda - Trailing Lambda 语法 二.Kotlin 中使用 Lambda 表达式替代对象表达式原理 1.Lambda 替换对象表达式 2.原理分析 3.示例分 ...

  2. python中如何创建类的对象_python面向对象中如何建立具体的对象?

    我们现在眼前所能看到的事物,都是具体的对象.很多小伙伴在面向对象中创建对象,其实都停留在对象名称的建立,计算机中并没有具体对象的描述属性.我们想要使用python中的类,建立的对象就需要是具体的.下面 ...

  3. python中函数type可以测试对象类型_Python中type函数type()显示对象的类型,dir函数dir()显示的是对象可用的方法。_学小易找答案...

    [判断题]new_word = word.upper() 这一语句表示将word这个变量的值全部小写并赋给新变量new_word. [单选题]All of the students have fini ...

  4. java 对象 序列化 文件中_如何将一个java对象序列化到文件里

    1.准备要序列化的类User (这个类必须实现Serializable接口,该接口只起到一个标记作用,没有任何抽象方法) package cn.qdm.ceshi; import java.io.Se ...

  5. 如何在html创建js对象,在js中使用createElement创建HTML对象和元素

    1.创建链接 var o = document.body; //创建链接 function createA(url,text) { var a = document.createElement(&qu ...

  6. 【基础篇】Kotlin第四讲-类、对象和接口

    类 类是逻辑组织的基本单元,类含有以下成分:依赖包,类名,构造方法,属性,成员方法,伴生对象,接口,父类等 类的构造方法的完整逻辑过程 先考虑主构造函数,当主构造函数不够用时,再引入从构造函数.最初的 ...

  7. Kotlin的基本数值类型问题:是对象?还是基本数据类型?

    前言 我们都知道Java中一直在说万物皆对象,面向对象编程,但基本数据类型却不是对象, 而Kotlin中的所有类型都是对象,包括基本数值类型和方法, 而数值对象的运行效率不如基本数据类型(具体请自行搜 ...

  8. es6中的变量 解构 对象 数组 函数

    文章目录 一.变量/常量声明 二.解构 2.1.数组解构 2.2对象解构 3.3.字符串解构 4.数值解构 5.布尔值解构 三.对象 3.1.对象简写 3.2API拓展 四.扩展运算符 五.数组 5. ...

  9. python里面的类和对象_Python中类和对象在内存中是如何保存?

    类以及类中的方法在内存中只有一份,而根据类创建的每一个对象都在内存中需要存一份,大致如下图: 如上图所示,根据类创建对象时,对象中除了封装 name 和 age 的值之外,还会保存一个类对象指针,该值 ...

最新文章

  1. 第十八课.支持向量机
  2. 日本那个大户又要抛了?
  3. 使用Flash读取COOKIE
  4. SICP 之斐波那契数
  5. MVC应用程序实现文件库(FlexPaper)
  6. python退出帮助系统help应该使用exit_python--help - tesion
  7. Spring对象绑定与类型转换
  8. hadoop 集群间数据迁移
  9. Linux oracle中文乱码的问题解决
  10. 转 SPI和API的区别
  11. oracle日期处理完全版
  12. 张至顺道长羽化登仙+说修行(道经每日清修)
  13. Visual Stdio 无法找到资源编译器DLL
  14. 快手之家(aardio.net) - 开头难
  15. 【解决方案】macOS 打开微信视频电话其他应用音量变小问题
  16. 小仙女手账的神仙句子
  17. 易基因项目文章|WGBS+RNA-seq揭示PM2.5引起男性生殖障碍的DNA甲基化调控机制
  18. ASO干货丨6招解决APP上架时内购频繁被拒问题
  19. 【Opencv项目实战】背景替换:动态背景移除与替换(cvzone+MediaPipe)
  20. 陕西万德软件有限公司

热门文章

  1. python中的json结构_python数据挖掘_Json结构分析
  2. mac解压rar命令_苹果mac电脑上很好用的免费压缩软件?ezip压缩软件分享
  3. java没有这样的元素异常_java – 没有这样的元素异常?
  4. qt项目中的某一个类的输出中文信息乱码,其它类中文输出正常
  5. Python基础教程学习目录 - Python入门教程
  6. php mysql增修删_PHP mysql PDO增、删、查、改
  7. sparkstreaming 读取mysql_第十篇|SparkStreaming手动维护Kafka Offset的几种方式
  8. 高并发编程_高并发编程系列:全面剖析Java并发编程之AQS的核心实现
  9. javaee 中遇到的jdk自带的异常(Exception)
  10. cfar恒虚警matlab实现,一种用于距离副瓣抑制的自适应恒虚警方法与流程