如何使用Java调Kotlin,今天开始使用Kotlin

2,328 阅读4分钟

如何使用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是很简单的。我列举了最典型的例子,用近似的方式所有的能被使用。