Android面试题之Kotlin和Java之间互操作

本文首发于公众号"AntDream",欢迎微信搜索"AntDream"或扫描文章底部二维码关注,和我一起每天进步一点点

互操作性和可空性

  • 要注意Java中所有类型都是可空的
  • String!表示平台数据类型
kotlin 复制代码
public class JavaTest {
        public String generateName() {
            return "name";
        }

        //可以用注解标注可空
        @Nullable
        public String generateNullString() {
            return null;
        }
    }



    fun main() {
        val result = JavaTest()
        //要用安全操作符来操作
        println(result?.generateName())
    }

类型映射

  • kotlin代码运行时,所有的映射类型都会重新映射回对应得Java类型

属性访问

  • 不需要调用相关setter和getter方法,可以用赋值语法来设置一个Java字段值
kotlin 复制代码
fun main() {
    val result = JavaTest()
    println(result.generateNullString())
    //对应的Java类中属性要实现get和set方法
    println(result.points)
}

@JvmName

  • 用来注解指定编译类的名字,方便Java调用

    //System.out.printf(HeroKt.sayHello()); //用JvmName改了名字后 System.out.printf(Hero.sayHello());

@JvmField

用@JvmField修饰后,Java中可以直接调用属性,不用getter方法

java 复制代码
class SpellBook {
    @JvmField
    val spells = "hello world"
}

public class JHaa {
    public static void main(String[] args) {
        SpellBook spellBook = new SpellBook();
        //没有@JvmField修饰时的调用方法
//        spellBook.getSpells();
        System.out.printf(spellBook.spells);
    }
}

@JvmOverloads

  • 协助产生kotlin函数的重载版本,设计一个可能会暴露给Java用户使用的API,记得使用 @JvmOverloads注解
kotlin 复制代码
class SpellBook {
        @JvmField
        val spells = "hello world"

        @JvmOverloads
        fun handleWithDefault(left:String = "left", right:String = "right"){
            println("you handle with $left and $right")
        }
    }

    public class JHaa {
        public static void main(String[] args) {
            SpellBook spellBook = new SpellBook();
    //        spellBook.getSpells();
            System.out.printf(spellBook.spells);
            //没有JvmOverloads注解时会报错,因为没有这个无参的重载函数版本
            spellBook.handleWithDefault();
        }
    }

@JvmStatic

  • @JvmField注解还能用来以静态方式提供伴生对象里定义的值
  • @JvmStatic注解的作用类似于@JvmField,允许你直接调用伴生对象的函数
less 复制代码
//互操作
    class SpellBook {
        @JvmField
        val spells = "hello world"

        @JvmOverloads
        fun handleWithDefault(left:String = "left", right:String = "right"){
            println("you handle with $left and $right")
        }

        companion object{
            @JvmField
            val MAX_SPELL_COUNT = 10
            @JvmStatic
            fun getSpell() = println("i am groot!")
        }
    }

    public class JHaa {
        public static void main(String[] args) {
            SpellBook spellBook = new SpellBook();
            //伴生对象属性的调用方式
    //        spellBook.Companion.getMAX_SPELL_COUNT();
            //伴生对象的属性加了@JvmField修饰以后
            int result = spellBook.MAX_SPELL_COUNT;

            spellBook.Companion.getSpell();
            //加了@JvmStatic修饰以后
            spellBook.getSpell();
        }
    }

@Throws

  • 抛出一个需要检查的指定异常,Java和kotlin有关的异常检查的差异让@Throws注解解决了,在编写供Java开发者调用的Kotlin API时,要考虑使用@Throws注解。这样用户就指定怎么正确处理任何异常了
  • kotlin中抛出的异常会被转换成Throwable,Java中会catch不了。加上@Throws注解就可以在Java中正常catch
kotlin 复制代码
class SpellBook {

        @Throws(IOException::class)
        fun funWithException() {
            throw IOException()
        }
    }

    public class JHaa {
        public static void main(String[] args) {
            SpellBook spellBook = new SpellBook();

            try {
                spellBook.funWithException();
                //没有Throws时,这里只能是Throwable,编译器也不会提示
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

函数类型操作

  • kotlin函数类型和匿名函数的简洁高效的语法因->操作符而实现,但Java8之前的JDK版本不支持lambda表达式。
  • 在Java里,kotlin函数类型使用FuctionN这样的名字的接口来表示,N代表值参的数目,这样的Function接口由23个,每一个都包含一个invoke函数,专门用于调用函数类型函数
typescript 复制代码
class SpellBook {
     
    val translator = { origin:String ->
        println(origin.lowercase().capitalize())
    }
}

public class JHaa {
    public static void main(String[] args) {
        SpellBook spellBook = new SpellBook();

        Function1<String, Unit> translator = spellBook.getTranslator();
        translator.invoke("TRACE");
}

欢迎关注我的公众号查看更多精彩文章!

相关推荐
小悟空7 小时前
[AI 生成] Flink 面试题
大数据·面试·flink
Jackilina_Stone8 小时前
【faiss】用于高效相似性搜索和聚类的C++库 | 源码详解与编译安装
android·linux·c++·编译·faiss
Sherry0079 小时前
CSS Grid 交互式指南(译)(下)
css·面试
一只毛驴9 小时前
浏览器中的事件冒泡,事件捕获,事件委托
前端·面试
一只叫煤球的猫9 小时前
你真的处理好 null 了吗?——11种常见但容易被忽视的空值处理方式
java·后端·面试
棒棒AIT9 小时前
mac 苹果电脑 Intel 芯片(Mac X86) 安卓虚拟机 Android模拟器 的救命稻草(下载安装指南)
android·游戏·macos·安卓·mac
KarrySmile9 小时前
Day04–链表–24. 两两交换链表中的节点,19. 删除链表的倒数第 N 个结点,面试题 02.07. 链表相交,142. 环形链表 II
算法·链表·面试·双指针法·虚拟头结点·环形链表
fishwheel9 小时前
Android:Reverse 实战 part 2 番外 IDA python
android·python·安全
一只毛驴10 小时前
谈谈浏览器的DOM事件-从0级到2级
前端·面试
在未来等你10 小时前
RabbitMQ面试精讲 Day 5:Virtual Host与权限控制
中间件·面试·消息队列·rabbitmq