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上提了个问题,结果很快就得到了回答。深切感受到了开发者的友善。”同一个世界,同一份代码“~ 感兴趣的同学可以去看下原答案。

相关文章
|
28天前
|
NoSQL Redis Kotlin
Kotlin教程笔记(17) - 属性代理
Kotlin教程笔记(17) - 属性代理
|
7天前
|
NoSQL Redis Kotlin
Kotlin - 属性代理
Kotlin - 属性代理
19 5
|
4天前
|
NoSQL Redis Kotlin
Kotlin - 属性代理
Kotlin - 属性代理
13 1
|
10天前
|
NoSQL Redis Kotlin
Kotlin - 属性代理
Kotlin - 属性代理
|
16天前
|
NoSQL Redis Kotlin
Kotlin教程笔记(17) - 属性代理
Kotlin教程笔记(17) - 属性代理
26 1
|
18天前
|
NoSQL Redis Kotlin
Kotlin教程笔记(17) - 属性代理
Kotlin教程笔记(17) - 属性代理
|
23天前
|
NoSQL Redis Kotlin
Kotlin教程笔记(17) - 属性代理
Kotlin教程笔记(17) - 属性代理
24 1
|
29天前
|
NoSQL Redis Kotlin
Kotlin教程笔记(17) - 属性代理
Kotlin教程笔记(17) - 属性代理
46 1
|
24天前
|
NoSQL Redis Kotlin
Kotlin教程笔记(17) - 属性代理
Kotlin教程笔记(17) - 属性代理
23 0
|
1月前
|
NoSQL Redis 开发者
Kotlin教程笔记(17) - 属性代理
本系列教程笔记详细讲解了Kotlin语法,适合需要深入了解Kotlin的开发者。对于希望快速掌握Kotlin语法的读者,建议查看“简洁”系列教程。本文重点介绍了Kotlin的属性代理功能,包括自定义属性代理、延迟属性 `lazy` 和可观察属性 `observable` 及 `vetoable`,并通过示例展示了它们的使用方法和原理。
24 0