Kotlin属性访问器与方法签名冲突的问题

简介: Kotlin属性访问器与方法签名冲突的问题

是什么问题?

如果你熟悉Java语言,那么你就会了解下面这个代码片段是不被允许的:

class SomeClass {
  public int getData() { /* ... */ } // 'getData()' clashes with 'getData()'; both methods have same erasure
  public String getData() { /* ... */ }
}

如果使用Kotlin编写,同样也是无法通过的:

class SomeClass {
  fun getData(): Int { /* ... */ } // Conflicting overloads ...
  fun getData(): String { /* ... */ }
}

但是,如果 SomeClass 这个类有一个属性是 data,那么情况就有点不一样:

// ok
class SomeClass {
  val data: Int = 0
  fun getData(): String { /* ... */ }
}

上面这种写法是被允许的,因为当我们使用 SomeClass 的实例区分别访问属性和方法时,访问方式并不一样:

val instance = SomeClass()
val intData : Int = instance.data
val stringData : String = instance.getData()

到这里,一切都合情合理。但如果我们看一下Kotlin反编译后的Java代码,会得到这样的结果:

public class SomeClass {
   private final int data;
   public final int getData() {
      return this.data;
   }
   @NotNull
   public final String getData() {
      return "";
   }
}

这么看就有点奇怪了,因为出现了上面示例中不允许的代码。

什么原因呢?

这个问题有三个点值得我们关注:

  1. 在Kotlin语法中,对属性的访问和对getXxx的访问并不是等价的。(虽然kotlin编译器可以自动把对Java getter的访问转换成属性访问)
  2. Java/Kotlin编译器的规则并不等于JVM的规则。
  3. 使用IDE查看Kotlin编译成的字节码反编译后的Java代码,不代表能通过Java编译器的编译。

1

第一个很好理解,对Kotlin编译器来说,这两者并不存在签名冲突,所以是合法的。

2

编译器的作用在于将文本代码编译成字节码,供JVM执行。”方法签名冲突“这个错误是编译器报告的错误,而不是JVM。

JVM在执行期间,对方法的调用是通过内存地址而不是方法签名,所以也就没有冲突的问题。

3

Kotlin会被编译成JVM字节码,而不是Java代码。反编译得到的Java代码不一定能通过Java编译器的编译。

总结

虽然Kotlin允许属性和getter方法同时存在,但是并不建议这样对属性和方法进行命名。首先命名上存在歧义;其次,如果在Java代码中调用这两个方法,Java编译器将会报错。因为Java编译器无法分data属性的getter方法和getData方法,导致无法通过编译。

看完原因再反过来看上面的问题,就会觉得这并不是个问题。但回想一下遇到这个问题时没有理解的原因,应该还是对Kotlin的理解出现了偏差。从Java转到Kotlin之后,很多时候还是会尝试从Java的角度来理解Kotlin,所以经常会忽略掉一个关键点:Kotlin基于JVM,而非Java

以这个角度作为前提,才能更 “Writing Kotlin the Kotlin Way” 吧。

虽然经常从 stackoverflow.com 上找到问题的答案,但是从来没有提问或者回答过,遇到这个问题(最开始以为和继承有关系)搜索无果后,尝试在stackoverflow上提了个问题,结果很快就得到了回答。深切感受到了开发者的友善。”同一个世界,同一份代码“~ 感兴趣的同学可以去看下原答案。

相关文章
|
2月前
|
安全 Java Kotlin
面试必备:Kotlin 线程同步的 N 种方法
面试必备:Kotlin 线程同步的 N 种方法
131 0
|
5天前
|
前端开发 Android开发 Kotlin
Kotlin小技巧之用Transformations.map方法转换LiveData
`Transformations.map`在Kotlin的Android开发中用于LiveData的数据转换,它在数据变化时自动转换并更新新LiveData。例如,从Int转为String。当原始LiveData更新时,转换后的LiveData也相应更新,适合MVVM架构。观察者可以订阅转换后的LiveData以更新UI。
14 2
|
18天前
|
设计模式 Java Kotlin
Kotlin中的委托、属性委托和延迟加载
Kotlin中的委托、属性委托和延迟加载
13 1
|
2月前
|
缓存 Android开发 数据安全/隐私保护
android开发,使用kotlin学习HTTP访问网络
android开发,使用kotlin学习HTTP访问网络
119 0
|
7月前
|
Kotlin
kotlin获取属性注解
kotlin获取属性注解
76 0
|
9月前
|
Cloud Native Go Kotlin
Kotlin 环境下解决属性初始化问题
Kotlin 环境下解决属性初始化问题
37 0
|
11月前
|
Java Kotlin
Kotlin中与Java互操作与可空性、类型映射、属性访问、@JvmOverloads、@JvmField、@JvmStatic、@Throws和函数类型操作详解
Kotlin中与Java互操作与可空性、类型映射、属性访问、@JvmOverloads、@JvmField、@JvmStatic、@Throws和函数类型操作详解
76 0
|
JSON 安全 Java
Kotlin学历之委托属性
Kotlin学历之委托属性
106 0
Kotlin学历之委托属性
|
Java Kotlin
Kotlin去掉UUID 横杠 方法
简简单单,一行代码搞定: val uuid = UUID.randomUUID().toString().replace("-", “”) 如果在Java中也可以用 UUID.randomUUID().toString().replaceAll("-", “”)
256 0
Kotlin去掉UUID 横杠 方法
|
安全 Java Kotlin
面试必备:Kotlin 线程同步的 N 种方法
面试的时候经常会被问及多线程同步的问题,例如,有 Task1、Task2 等多个并行任务,如何等待全部执行完成后执行 Task3?
608 0