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


猜你喜欢
- 本文实例讲述了C#实现两接口中同名方法。分享给大家供大家参考。具体分析如下:对于一个类实现两个接口,而这两个接口又有同名方法,C#中的处理方
- java异常分为两大类,Checked异常和Runtime异常,Checked异常都是在编译阶段可以被处理的异常。Checked异常和Run
- Spring AOP proxyTargetClass的行为要点列表形式proxyTargetClasstrue目标对象实现了接口 – 使用
- 本文实例为大家分享了C语言实现两个矩阵相乘的具体代码,供大家参考,具体内容如下程序功能:实现两个矩阵相乘的C语言程序,并将其输出代码如下:#
- 本文实例讲述了C#实现TCP连接信息统计的方法。分享给大家供大家参考。具体实现方法如下:using System;using System.
- 本文实例分析了C#中var关键字用法。分享给大家供大家参考。具体方法如下:C#关键字是伴随着.NET 3.5以后,伴随着匿名函数、LINQ而
- 向shell提供命令非常简单,需要学习的注解很少。该命令的实现风格与使用依赖注入的应用程序的开发类相同,您可以利用Spring容器的所有特性
- 本文实例讲述了Java Swing中JDialog实现用户登陆UI。分享给大家供大家参考,具体如下:JDialog是一种对话框组件,它常常与
- 方法重载概述方法重载指同一个类中定义的多个方法之间的关系,满足下列条件的多个方法互相构成重载* 多个方法在同一个类中* 多个放方法具有相同方
- Android Studio第一次启动的Fetching android sdk component information的问题1)进入刚
- 一、项目运行环境配置:Jdk1.8 + Tomcat8.5 + Mysql + HBuilderX(Webstorm也行)+ Eclispe
- 本文实例讲述了C语言实现的猴子分桃问题算法。分享给大家供大家参考,具体如下:问题:海滩上有一堆桃子,五只猴子来分。第一只猴子把这堆桃子凭据分
- @RequestBody与post请求的关系@RequestBody主要用来接收前端传递给后端的json字符串中的数据的(请求体中的数据的)
- 前言WebSocket 使得客户端和服务器之间的数据交换变得更加简单,允许服务端主动向客户端推送数据。在 WebSocket API 中,浏
- 前言不知道你是否参加过拼多多上邀请微信好友砍价功能,这个功能实现首先需要考虑的就是获取微信用户的信息。获取用户信息就是获取公众号下微信用户的
- SnowFlake 算法,是 Twitter 开源的分布式 id 生成算法。其核心思想就是:使用一个 64 bit 的 long 型的数字作
- 本文汇总了C#启动外部程序的几种常用方法,非常具有实用价值,主要包括如下几种方法:1. 启动外部程序,不等待其退出。2. 启动外部程序,等待
- @FeignClient()注解的使用由于SpringCloud采用分布式微服务架构,难免在各个子模块下存在模块方法互相调用的情况。比如se
- SpringSecurity 框架简介Spring 是非常流行和成功的 Java 应用开发框架,Spring Security 正是 Spr
- 问题介绍测试服务器突然无法连接,ssh登录不上。只有重启才能解决。重启一天后,又连接不上了。于是有了下面的排查过程,最终发现是有个java程