本文链接
本文结合自己的感受,做一下简单的翻译。原文作者也是《Kotlin for Android developer》的作者。此译文供大家学习参考之用。
如何使用Java调Kotlin:今天开始使用Kotlin
Kotlin 一个极好的点就是完全和Java融为一体。这个意味着,即使你的应用代码是使用Java写的,你能用Kotlin创建一个类,然后使用Java调用它不产生任何问题。
这至少给你带来2个潜在优势:
你能在Java项目中使用Kotlin:在任何你已经开始的项目中,你可以决定使用Kotlin写新的代码。你也可以用Java代码调用它。
如果你在使用Kotlin的时候有障碍,那这部分可以用Java代替:许多人问我,如果你在Android上面用Kotlin不足以做某些东西的时候。理论上,所有的都可以做,实际中没有办法知道(没有人在Android上用Kotlin做所有的事)。现实这不是问题,如果你不用Kotlin做,可以换成用Java来实现这部分内容。
今天我们来看看 他们如何相互并存工作, 当你用Java调用Kotlin时,Kotlin代码是怎样的。
从Java中调用Kotlin:包级别函数
在Kotlin中,函数不必一定在类中的,但是Java中没有这样的使用方式。我们要怎么样调用这样的函数呢?想象一下我们有一个文件utils.kt,看上去如下:
fun logD(message: String) {
Log.d("", message)
}
fun logE(message: String) {
Log.e("", message)
}
在Java中,我们通过一个来访问他们,这个类叫UtilsKt,有一些静态方法:
UtilsKt.logD("Debug");
UtilsKt.logE("Error");
如果你看过之前分享的主题,我喜欢扩展函数。那他们在Java中看起来是怎样的?想象我们有如下:
fun ViewGroup.inflate(resId: Int, attachToRoot: Boolean = false): View {
return LayoutInflater.from(context).inflate(resId, this, attachToRoot)
}
注意:虽然他们会出现在一些地方,我没有明确评论它。这些函数参数可能默认值。这意味着,即使我们不具体声明他们,他们也会使用具体定义的默认值。这防止我们像使用Java一样的重写方法。
这个扩展函数使用在 ViewGroup上。它获得layout而且使用父视图填充它。
如果我们在Java中使用它,我们要怎么样用?
View v = UtilsKt.inflate(parent, R.layout.view_item, false);
如你看到的,使用这个函数的对象本身在Java中被作为函数的第一个参数,并且可选参数变成强制的,Java不能使用默认值。
函数重载
如果你要生成和在Java中重载一致的,你可以用@JvmOverloads注解那个函数。这样,你在Java中不必强制使用false实参。
@JvmOverloads
fun ViewGroup.inflate(resId: Int, attachToRoot: Boolean = false): View {
return LayoutInflater.from(context).inflate(resId, this, attachToRoot)
}
View v = UtilsKt.inflate(parent, R.layout.view_item);
如果你想在Java调用的时候情愿用自己定义的类名的话,加你可以用一个注解修改它。在这个文件utils.kt中,在package之前加上:
@file:JvmName("AndroidUtils")
然后现在这个类在Java被命名如下:
AndroidUtils.logD("Debug");
AndroidUtils.logE("Error");
View v = AndroidUtils.inflate(parent, R.layout.view_item, false);
实例和静态属性
在Java中我们使用属性保存状态。他们可以是实例属性,即每个对象自己,或者静态的(所有类实例共享它们的)。如果我们试着在Kotlin中找对应的,应该是属性和companion对象。下面是类似的类:
class App : Application() {
val appHelper = AppHelper()
companion object {
lateinit var instance: App
}
override fun onCreate() {
super.onCreate()
instance = this
}
}
这个在Java中是如何工作的?你可以用getters和setters的方法,简单的访问companion对象类似静态属性:
AppHelper helper = App.instance.getAppHelper();
你能看到编译器只能这样, 当用val修饰时,它只在Java中生成getter 方法,如果用var修饰,我们才能看到setter 方法。访问实例时的自动工作,因为它使用了lateinit 注解,在Kotlin使用它存储状态的时候自动初始化。但想象我们创建一个常量。
companion object {
lateinit var instance: App
val CONSTANT = 27
}
看到没,你不能直接访问它,你要通过一个Companion 内部类来访问:
KotlinTest.Companion.getCONSTANT()
这看起来不怎么好。如果要像Java一样简单的暴露一个静态属性,你需要使用一个新注解:
@JvmField val CONSTANT = 27
现在你能从Java代码中使用它:
int c = App.CONSTANT;
如果你在companion对象中有函数,他们能用@JvmStatic注解来转换成静态方法。
总结
你能看到从Java中调用Kotlin是很简单的。我列举了最典型的例子,用近似的方式所有的能被使用。