欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

在Java與Kotlin之間如何進行互操作詳解

 更新時間:2019年12月04日 08:43:05   作者:李四爺  
這篇文章主要給大家介紹了關于在Java和Kotlin之間如何進行互操作的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧

前言

目前kotlin是谷歌首推的開發(fā)Android的語言,但由于歷史原因,我們絕大部分項目依舊還是以Java為主的,也就是說存在Java和Kotlin兩種語言同時開發(fā)的情況。

有人會說把老項目全部翻譯成Kotlin,的確可以怎么做,但是成本還是挺大的。我們只能一點一點慢慢的向kotlin語言遷移。

那么在遷移的過程中就避免不了Java和Kotlin相互調用的情況。即Kotlin調用Java或者Java調用Kotlin。下面我們就來具體看下兩者之間相互操作的一些解決方案。

kotlin調用java

可空性(Nullability)

Java默認有數值可空性而kotlin沒有,所以在調用Java的方法的時候不知道會不會收到空值。
所以我們在Kotlin中調用Java的時候需要添加 ?或者 !來告訴Kotlin有可能出現空值。

比如這里有一個Java方法,接受一組字符串后返回一組做字符串:

public Set<String> toSet(Collection<String> elements){
 //TODO
}

那么Kotlin在調用的時候是不能確定輸入和輸出是否可為空的。就需要使用?或者 !來​輔助判斷。​

為了方便Kotlin調用,我們通常使用 @NotNull 注解來標識Java代碼的非原始參數、字段、返回值。

@NotNull
Set<@NotNull String> toSet(@NotNull Collection<@NotNull String> elements){
 //TODO
}

這個Kotlin在調用的時候就明確知道不能為空,這里我們使用的是jetBrain的 @NotNull注解,當然還有其他選擇,如下圖:

這里還是推薦使用JetBrain或者Android的注解。

前綴屬性:(getter、setter)

如果是使用Java bean,那么我們在Kotlin中調用就沒有什么問題。

如果你的空參數方法是以get開頭的,那么Kotlin就知道這是getter,就可以通過屬性名來訪問它。

相同的如果是由set開頭的單一參數方法,那么Kotlin就知道這是setter,就通過屬性名直接賦值。

當然is的工作原理也是和它們類似的。

我們定義一個Java bean:

class User {
 
 private String name;
 private int age;

 public String getName() {
  return name;
 }

 public void setName(String name) {
  this.name = name;
 }

 public int getAge() {
  return age;
 }

 public void setAge(int age) {
  this.age = age;
 }
}

Kotlin中訪問

val user = User()
user.name = "四爺" //賦值
val age = user.age //獲取age字段值

關鍵字(keywords)

kotlin中有很多系統(tǒng)定義的關鍵字如 fun is in objects、typeof、val、var、when、typealias等。

這些關鍵字在java是可以被使用的,但是在kotlin卻是不行的。

函數或者參數使用了這些關鍵字,那么kotlin在調用的時候會出現一些問題,比如Java中定義了一個方法名叫 is 的方法。那么在Kotlin中直接調用就會報錯。

那么最簡單的方法就是重命名Java方法,但如果調用的是三方庫的方法,就很難去重命名了。
所以我們另一種解決方式是在Kotlin調用java方法的時候加上 `` 反引號來使用。

Utils.`is`()

但是我們如果能重命名還是重命名,以防止代碼出現太多的符號。

避免在任何擴展方法和擴展屬性上使用Any

運算符重載(operator Overloading)

在Java不存在運算符重載,而kotlin有。比如:

a+b => a.plus(b)

在kotlin中將運算符 + 翻譯為了方法 plus。

如果在Java中使用了同樣的方法名稱,比如 加(plus)、 減(minus)或者其他運算符名稱,那么請務必確保他們與運算符兼容,避免意外調用他們。

Java調用Kotlin

JvmName & JvmMultifileClass

當我們在遷移的時候會將Java的工具類翻譯為Kotlin拓展函數或者頂層函數。但是這樣處理之后,在Java文件中是無法直接調用的,此時我們需要加注解 @file:JvmName(“文件名稱”):

Ext.kt文件

@file:JvmName("ExtUtils")
package com.demo.javaAndKotlin

fun a(): String {
 ...
}

fun b(): String {
 ...
}

這里我們將名稱命名為ExtUtils。此外,我們可能還有其他的頂層函數或者擴展函數。按照上面這種方式我們也可以指定一個其他的名稱,但是如果我們也想使用ExtUtils這個名稱的時候會報錯:

Duplicate JVM class name

此時我們需要在不同的文件中加入新的注解 @file:JvmMultifileClass 。意思是將所有的文件合并到一個新的名稱為ExtUtils文件中。

ExtOther.kt文件

@file:JvmMultifileClass
@file:JvmName("ExtUtils")
package com.demo.javaAndKotlin

fun c(str: Any): String {
 ...
}

我們在Ext.kt文件中也加入@file:JvmMultifileClass注解,我們就可以在Java文件中直接使用ExtUtils來調用 a(),b(),c()方法了。

JvmField

在 kotlin中我們使用的數據類即 data class 是不需要指定getter和setter的,可以直接通過字段名來訪問它們。但是如果是在Java文件中調用data class依舊是需要使用getter和setter方法進行調用的。這里我們是可以修改他們的,那就是使用 @JvmField 注解,通過注解,可以直接將字段暴露出去進行訪問。​

data class Person(

 @JvmField var name: String,
 @JvmField var age: Int
)

//java中調用
Person person = new Person("",1);
person.name = "";
person.age = 10;

但是也有例外就是lateinit修飾的字段會自動暴露,無需指定@JvmField注解。還有const修飾的字段也是一樣會自動暴露。

另外,如果我們想在Java中調用setName的時候修改這個屬性名稱不叫setName,這里我們需要使用@set:JvmName 注解。同理修改getName使用@get:JvmName 。需要注意的是,指定了@set:JvmName或者@get:JvmName注解后不需要在指定@JvmField了。

data class Person(

 @set:JvmName("changeName")
 var name: String,
 @JvmField var age: Int,
 @get:JvmName("likesPink")
 var likesPink: Boolean
){
 lateinit var address:String
}

JvmStatic

當我們將Java文件的靜態(tài)方法遷移到Kotlin中時,我們會將其放在 companion object中,但是這樣處理之后在Java文件中無法直接調用,得通過companion對象實例方法來調用。

class MyService {
 internal fun doWork() {
  ...
 }

 companion object {
  fun schedule(context: Context) {
   ...
  }
 }
}

//在Java中調用
MyService.Companion.schedule(this);

幸運的是Kotlin提供了 @JvmStatic 注解。他會讓Kotlin在編譯器完成類封裝后生成一個靜態(tài)方法。

class MyService {
 internal fun doWork() {
  ...
 }

 companion object {
   @JvmStatic
  fun schedule(context: Context) {
   ...
  }
 }
}

//在Java中調用
MyService.schedule(this);

JvmOverloads

在Kotlin中我們可以給函數的參數設置默認值,即默認參數。但是這個功能在Java中是沒有的。如果不做任何處理,那么在Java中調用函數的時候,就必須每個參數都要傳入。那么我們設置的默認參數就沒有任何意義了。

所以,Kotlin給我們提供了 @JvmOverloads注解,使用這個注解后,會讓Kotlin編譯器按照從左向右的順序依次為每一個可選參數生成重載。

@JvmOverloads
fun Bitmap.resize(width: Int, height: Int = 200) {

}

//java調用
ExtUtils.resize(bitmap,100);

這里我們在Kotlin中很容易就理解了Bitmap.resize方法的含義,但是ExtUtils.resize這樣調用的時候,方法名不夠明確。所以我們可以使用@JvmName注解來指定名稱。

@JvmName("resizeBitmap")
@JvmOverloads
fun Bitmap.resize(width: Int, height: Int = 200) {

}
//java調用
ExtUtils.resizeBitmap(bitmap,100);

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

相關文章

  • Spring Boot 整合持久層之JdbcTemplate

    Spring Boot 整合持久層之JdbcTemplate

    持久層是 Java EE 中訪問數據庫的核心操作,Spring Boot 中對常見的持久層框架都提供了自動化配置,例如 JdbcTemplate 、 JPA 等,Mybatis 的自動化配置則是 Mybatis 官方提供的
    2022-08-08
  • 為什么不推薦使用BeanUtils屬性轉換工具示例詳解

    為什么不推薦使用BeanUtils屬性轉換工具示例詳解

    這篇文章主要介紹了為什么不推薦使用BeanUtils屬性轉換工具,本文通過示例代碼給大家詳細介紹,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-07-07
  • Java動態(tài)添加view的方法

    Java動態(tài)添加view的方法

    這篇文章主要介紹了Java動態(tài)添加view的方法,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2017-01-01
  • Java SPI的簡單小實例

    Java SPI的簡單小實例

    這篇文章主要介紹了Java SPI的簡單小實例,文中講解非常細致,代碼幫助大家更好的理解和學習,感興趣的朋友可以了解下
    2020-07-07
  • 解決idea中yml文件圖標問題及自動提示失效的情況

    解決idea中yml文件圖標問題及自動提示失效的情況

    這篇文章主要介紹了解決idea中yml文件圖標問題及自動提示失效的情況,具有很好的價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01
  • springmvc視圖解析流程代碼實例

    springmvc視圖解析流程代碼實例

    這篇文章主要介紹了springmvc視圖解析流程代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-01-01
  • Java中StringUtils工具類進行String為空的判斷解析

    Java中StringUtils工具類進行String為空的判斷解析

    這篇文章主要介紹了Java中StringUtils工具類進行String為空的判斷解析,具有一定借鑒價值,需要的朋友可以參考下
    2018-01-01
  • Java8中常見函數式接口的使用示例詳解

    Java8中常見函數式接口的使用示例詳解

    在 Java 8 中,函數式接口是一個關鍵的特性,它們允許將方法作為參數傳遞或返回類型,本文為大家整理了一些常見的函數式接口的使用,希望對大家有所幫助
    2023-12-12
  • java實現背單詞程序

    java實現背單詞程序

    這篇文章主要為大家詳細介紹了java實現背單詞程序,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • Java中線程的基本方法使用技巧

    Java中線程的基本方法使用技巧

    這篇文章主要介紹了Java中線程的基本方法使用技巧,需要的朋友可以參考下
    2017-09-09

最新評論