Kotlin中的对象表达式和对象声明的具体使用

作者:Hunter_Arley 时间:2022-05-31 04:08:18 

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<String>) {
 //指定一个父类型(接口)的对象表达式
 var ob1 = object : Outputable {
   override fun output(msg: String) {
     for (i in 1..6) {
       println("<h${i}>${msg}</h${i}>")
     }
   }
 }
 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()
}

输出结果:

<h1>随便输出点什么吧</h1>
<h2>随便输出点什么吧</h2>
<h3>随便输出点什么吧</h3>
<h4>随便输出点什么吧</h4>
<h5>随便输出点什么吧</h5>
<h6>随便输出点什么吧</h6>
-----------------------------------------------
初始化块
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<String>) {
 ObjectExprType().test()
}

输出结果:

Kotlin
Java

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

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


fun main(args: Array<String>) {
 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("<h${i}>${msg}</h${i}>")
   }
 }
}

//指定零个父类型的对象表达式
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<String>) {

MyObject1.output("一起来学Kotlin")
 println("-----------------------------------------------")
 println(MyObject2.name)
 MyObject2.test()
 println("-----------------------------------------------")
 println(MyObject3.name)
 MyObject3.output("Kotlin真不错")
 MyObject3.printInfo()
}

输出结果:

<h1>一起来学Kotlin</h1>
<h2>一起来学Kotlin</h2>
<h3>一起来学Kotlin</h3>
<h4>一起来学Kotlin</h4>
<h5>一起来学Kotlin</h5>
<h6>一起来学Kotlin</h6>
-----------------------------------------------
初始化块
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("<h${i}>${msg}</h${i}>")
     }
   }
 }
}

fun main(args: Array<String>) {
 //使用伴生对象所在的类调用伴生对象的方法
 MyClass.output("Kotlin必须学")
 println(MyClass.name)
}

输出结果:

<h1>Kotlin必须学</h1>
<h2>Kotlin必须学</h2>
<h3>Kotlin必须学</h3>
<h4>Kotlin必须学</h4>
<h5>Kotlin必须学</h5>
<h6>Kotlin必须学</h6>
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("<h${i}>${msg}</h${i}>")
     }
   }
 }
}

//为伴生对象扩展方法
fun MyClass.Companion.test() {
 println("为伴生对象扩展的方法")
}

val MyClass.Companion.foo
 get() = "为伴生对象扩展的属性"

fun main(args: Array<String>) {
 //使用伴生对象所在的类调用伴生对象的方法
 MyClass.output("Kotlin必须学")
 println(MyClass.name)
 //通过伴生对象所在的类调用为伴生对象扩展的成员
 MyClass.test()
 println(MyClass.foo)

}

输出结果:

<h1>Kotlin必须学</h1>
<h2>Kotlin必须学</h2>
<h3>Kotlin必须学</h3>
<h4>Kotlin必须学</h4>
<h5>Kotlin必须学</h5>
<h6>Kotlin必须学</h6>
name属性值
为伴生对象扩展的方法
为伴生对象扩展的属性

来源:https://www.jianshu.com/p/278bfc6a38fc

标签:Kotlin,对象表达式,对象声明
0
投稿

猜你喜欢

  • java新手入门——String类详解

    2022-02-23 08:17:26
  • Java循环对bean的属性进行赋值的实现

    2023-01-27 10:18:05
  • 源码解析JDK 1.8 中的 Map.merge()

    2023-11-16 23:49:25
  • C#判断一天、一年已经过了百分之多少的方法

    2022-07-16 15:23:11
  • jvm调优的几种场景(小结)

    2023-04-11 18:37:04
  • Android开发中amera2 Preview使用详解

    2023-11-09 20:17:56
  • Java多线程之条件对象Condition

    2021-09-25 15:15:30
  • 使用 C# 下载文件的多种方法小结

    2023-11-08 06:59:37
  • JavaWeb实现学生信息管理系统(2)

    2022-03-04 02:31:12
  • 详解用RxJava实现事件总线(Event Bus)

    2022-02-13 16:43:18
  • java日期时间操作工具类

    2022-09-03 13:04:58
  • Android实现截屏方式整理(总结)

    2023-12-07 05:10:18
  • C++实现哈夫曼树算法

    2023-05-21 01:03:46
  • 详解JAVA 线程-线程的状态有哪些?它是如何工作的?

    2023-11-27 03:33:09
  • 详解Mybatis多参数传递入参四种处理方式

    2022-12-27 15:18:20
  • Spring MVC整合 freemarker及使用方法

    2022-06-06 16:41:41
  • C语言数据结构之二叉树的非递归后序遍历算法

    2021-12-23 07:10:52
  • Android使用Gallery实现照片拖动的特效

    2022-12-31 23:04:22
  • C#在复杂多线程环境下使用读写锁同步写入文件

    2021-06-16 03:45:53
  • maven多profile 打包下 -P参和-D参数的实现

    2023-03-22 08:25:08
  • asp之家 软件编程 m.aspxhome.com