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

前端之家收集整理的这篇文章主要介绍了Kotlin中的对象表达式和对象声明的具体使用前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

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

一、对象表达式

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

  1. object [: 0~N父类型]{
  2. //对象表达式的类体部分
  3. }

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

  1. package `0705`
  2.  
  3. interface Outputable {
  4. fun output(msg: String)
  5. }
  6.  
  7. abstract class Product(var price: Double) {
  8. abstract val name: String
  9. abstract fun printInfo()
  10. }
  11.  
  12. fun main(args: Array<String>) {
  13. //指定一个父类型(接口)的对象表达式
  14. var ob1 = object : Outputable {
  15. override fun output(msg: String) {
  16. for (i in 1..6) {
  17. println("<h${i}>${msg}</h${i}>")
  18. }
  19. }
  20. }
  21. ob1.output("随便输出点什么吧")
  22. println("-----------------------------------------------")
  23. //指定零个父类型的对象表达式
  24. var ob2 = object {
  25. //初始化块
  26. init {
  27. println("初始化块")
  28. }
  29.  
  30. //属性
  31. var name = "Kotlin"
  32.  
  33. //方法
  34. fun test() {
  35. println("test方法")
  36. }
  37.  
  38. //只能包含内部类,不可以包含嵌套类
  39. inner class Inner
  40. }
  41. println(ob2.name)
  42. ob2.test()
  43. println("-----------------------------------------------")
  44. //指定两个父类型的对象表达式
  45. var ob3 = object : Outputable,Product(1.23) {
  46. override fun output(msg: String) {
  47. println("输出信息:${msg}")
  48. }
  49.  
  50. override val name: String
  51. get() = "激光打印机"
  52.  
  53. override fun printInfo() {
  54. println("高速极光打印机们支持自动双面打印!")
  55. }
  56. }
  57. println(ob3.name)
  58. ob3.output("Kotlin慢慢学")
  59. ob3.printInfo()
  60. }

输出结果:

<h1>随便输出点什么吧</h1>
<h2>随便输出点什么吧</h2>
<h3>随便输出点什么吧</h3>
<h4>随便输出点什么吧</h4>
<h5>随便输出点什么吧</h5>
<h6>随便输出点什么吧</h6>
-----------------------------------------------
初始化块
Kotlin
test方法
-----------------------------------------------
激光打印机
输出信息:Kotlin慢慢学
高速极光打印机们支持自动双面打印!

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

  1. package `0705`
  2.  
  3. class ObjectExprType {
  4. private val ob1 = object {
  5. val name: String = "Kotlin"
  6. }
  7. internal val ob2 = object {
  8. val name: String = "Kotlin"
  9. }
  10. private fun privateBar()=object {
  11. val name:String="Java"
  12. }
  13. fun publicBar()=object {
  14. val name:String="Java"
  15. }
  16. fun test(){
  17. //ob1是private对象表达式,编译器可识别它的真实类型
  18. println(ob1.name)
  19. //ob2是非private对象表达式,编译器当它是Any类型
  20. // println(ob2.name)
  21. //privateBar是private函数,编译器可识别它返回的对象表达式的真实类型
  22. println(privateBar().name)
  23. //publicBar是非private函数,编译器将它返回的对象表达式当成Any类型
  24. // println(publicBar().name)
  25. }
  26. }
  27.  
  28. fun main(args: Array<String>) {
  29. ObjectExprType().test()
  30. }

输出结果:

Kotlin
Java

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

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

  1. fun main(args: Array<String>) {
  2. var a = 20
  3. var obj = object {
  4.  
  5. fun change() {
  6. println("change()方法修改变量a的值")
  7. a++
  8. }
  9. }
  10. obj.change()
  11. println(a)
  12. }

输出结果:

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

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

二、对象声明和单例模式

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

  1. object ObjectName [: 0~N父类型]{
  2. //对象表达式的类体部分
  3. }

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

两者还有如下区别:

  1. package `0705`
  2.  
  3. interface Outputable {
  4. fun output(msg: String)
  5. }
  6.  
  7. abstract class Product(var price: Double) {
  8. abstract val name: String
  9. abstract fun printInfo()
  10. }
  11.  
  12. //指定一个父类型的对象表达式
  13. object MyObject1 : Outputable {
  14. override fun output(msg: String) {
  15. for (i in 1..6) {
  16. println("<h${i}>${msg}</h${i}>")
  17. }
  18. }
  19. }
  20.  
  21. //指定零个父类型的对象表达式
  22. object MyObject2 {
  23. //初始化块
  24. init {
  25. println("初始化块")
  26. }
  27.  
  28. //属性
  29. var name = "Kotlin"
  30.  
  31. //方法
  32. fun test() {
  33. println("test方法")
  34. }
  35.  
  36. //只能包含嵌套类,不可以包含内部类
  37. class Inner
  38. }
  39.  
  40. //指定两个父类型的对象表达式
  41. object MyObject3 : Outputable,Product(1.23) {
  42. override fun output(msg: String) {
  43. println("输出信息:${msg}")
  44. }
  45.  
  46. override val name: String
  47. get() = "激光打印机"
  48.  
  49. override fun printInfo() {
  50. println("高速极光打印机们支持自动双面打印!")
  51. }
  52. }
  53.  
  54. fun main(args: Array<String>) {
  55.  
  56. MyObject1.output("一起来学Kotlin")
  57. println("-----------------------------------------------")
  58. println(MyObject2.name)
  59. MyObject2.test()
  60. println("-----------------------------------------------")
  61. println(MyObject3.name)
  62. MyObject3.output("Kotlin真不错")
  63. MyObject3.printInfo()
  64. }

输出结果:

<h1>一起来学Kotlin</h1>
<h2>一起来学Kotlin</h2>
<h3>一起来学Kotlin</h3>
<h4>一起来学Kotlin</h4>
<h5>一起来学Kotlin</h5>
<h6>一起来学Kotlin</h6>
-----------------------------------------------
初始化块
Kotlin
test方法
-----------------------------------------------
激光打印机
输出信息:Kotlin真不错
高速极光打印机们支持自动双面打印!

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

三、伴生对象和静态成员

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

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

  1. package `0705`
  2.  
  3. interface CompanionTest {
  4. fun output(msg: String)
  5. }
  6.  
  7. class MyClass {
  8. //使用companion修饰的伴生对象
  9. companion object MyObject1 : CompanionTest {
  10. val name = "name属性值"
  11. override fun output(msg: String) {
  12. for (i in 1..6) {
  13. println("<h${i}>${msg}</h${i}>")
  14. }
  15. }
  16. }
  17. }
  18.  
  19. fun main(args: Array<String>) {
  20. //使用伴生对象所在的类调用伴生对象的方法
  21. MyClass.output("Kotlin必须学")
  22. println(MyClass.name)
  23. }

输出结果:

<h1>Kotlin必须学</h1>
<h2>Kotlin必须学</h2>
<h3>Kotlin必须学</h3>
<h4>Kotlin必须学</h4>
<h5>Kotlin必须学</h5>
<h6>Kotlin必须学</h6>
name属性

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

四、伴生对象的扩展

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

  1. package `0705`
  2.  
  3. interface CompanionTest {
  4. fun output(msg: String)
  5. }
  6.  
  7. class MyClass {
  8. //使用companion修饰的伴生对象
  9. companion object : CompanionTest {
  10. val name = "name属性值"
  11. override fun output(msg: String) {
  12. for (i in 1..6) {
  13. println("<h${i}>${msg}</h${i}>")
  14. }
  15. }
  16. }
  17. }
  18.  
  19. //为伴生对象扩展方法
  20. fun MyClass.Companion.test() {
  21. println("为伴生对象扩展的方法")
  22. }
  23.  
  24. val MyClass.Companion.foo
  25. get() = "为伴生对象扩展的属性"
  26.  
  27. fun main(args: Array<String>) {
  28. //使用伴生对象所在的类调用伴生对象的方法
  29. MyClass.output("Kotlin必须学")
  30. println(MyClass.name)
  31. //通过伴生对象所在的类调用为伴生对象扩展的成员
  32. MyClass.test()
  33. println(MyClass.foo)
  34.  
  35. }

输出结果:

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

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

猜你在找的Android相关文章