当前位置: 首页 > news >正文

Kotlin和Java区别

哈哈哈,前段时间,面试的时候,突然问到我Kotlin和Java的区别,一下子把我问懵逼了,确实没遇到问这个的,想了下,说了下Kotlin的编译时空检查机制,代码更简洁,很多封装好的API可以直接调用,Kotlin有Jetpack全家桶,有协程,有lateinit和by lazy懒加载机制等等,后面着重问我协程去了,但是我知道我也有很多没有答上。

今天自己做个总结:

1. 语法简洁性

🔹 Kotlin 代码通常比 Java 更简洁,减少了模板代码(boilerplate)。

🔸 Java 示例(获取 List 的大小并遍历):

List<String> list = Arrays.asList("A", "B", "C");
for (String item : list) {System.out.println(item);
}

Kotlin

val list = listOf("A", "B", "C")
list.forEach { println(it) }

2. Null 安全

🔹 Java 中 NullPointerException (NPE) 是常见错误

String name = null; 
System.out.println(name.length()); // 运行时崩溃:NullPointerException

🔹 Kotlin 通过可空类型和安全调用避免 NPE:

var name: String? = null
println(name?.length) // 安全调用,避免 NPE
  • ? 允许 null
  • ?. 安全访问,避免 NPE
  • !! 强制非空,可能引发异常,尽量少用。

3. 数据类 (Data Class)

🔹 Java 需要大量代码来定义 POJO(数据类):

public class User {private String name;private int age;public User(String name, int age) { this.name = name; this.age = age; }public String getName() { return name; }public int getAge() { return age; }public void setName(String name) { this.name = name; }public void setAge(int age) { this.age = age; }@Overridepublic String toString() { return "User{name='" + name + "', age=" + age + "}"; }
}

🔹 Kotlin 只需一行代码:

data class User(val name: String, val age: Int)
  • 自动生成 getter/settertoString()equals()hashCode()

data class的详细介绍和区别:Serializable,Parcelable和data class的区别_data class parcelable-CSDN博客文章浏览阅读992次,点赞24次,收藏26次。序列化是将对象的状态(属性数据)转换为字节流或其他可存储或传输的格式的过程。主要作用存储:将对象保存到文件或数据库中。传输:在网络中传输对象,比如在客户端与服务器之间传递数据。缓存:将对象转化为可恢复的格式,便于后续恢复使用。序列化后7. 什么是反序列化(Deserialization)?反序列化是将序列化后的字节流(或存储格式)重新转换回原始对象的过程。主要作用从存储或传输的格式中重建对象。恢复数据到应用中,便于程序继续使用。8.序列化与反序列化的用途网络传输。_data class parcelable https://blog.csdn.net/LoveFHM/article/details/143875848?spm=1001.2014.3001.5502

4. 扩展函数 (Extension Functions)

🔹 Java 需要创建工具类来扩展已有类的功能

public class StringUtils {public static String capitalize(String str) {return str.substring(0, 1).toUpperCase() + str.substring(1);}
}
String result = StringUtils.capitalize("hello");

🔹 Kotlin 直接扩展类的方法

fun String.capitalizeFirst(): String = this.replaceFirstChar { it.uppercaseChar() }
val result = "hello".capitalizeFirst()

5. 函数式编程

Kotlin 支持 高阶函数Lambda 表达式,让代码更优雅。

🔹 Java 的匿名内部类:

button.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {System.out.println("Clicked");}
});

🔹 Kotlin 的 Lambda 表达式:

button.setOnClickListener { println("Clicked") }

6. 协程 vs. Java 线程

🔹 Java 使用线程管理并发,代码较复杂:

new Thread(new Runnable() {@Overridepublic void run() {System.out.println("线程运行中...");}
}).start();

🔹 Kotlin 协程更高效,且不会阻塞线程

GlobalScope.launch {println("协程运行中...")
}
  • 协程比 Java 线程更轻量,可以高效处理并发任务。管理起来也很轻松,可以和生命周期绑定

Kotlin 协程(一)协程的定义及基础使用_kotlin 协程使用-CSDN博客

7. Smart Cast(智能类型转换)

🔹 Java 需要显式类型转换

Object obj = "Hello";
if (obj instanceof String) {String str = (String) obj; // 需要手动转换
}

🔹 Kotlin 自动类型转换

val obj: Any = "Hello"
if (obj is String) {println(obj.length) // Kotlin 自动转换,无需 `(String) obj`
}
  • 智能类型推断:Kotlin 可以自动推断变量类型,无需显式声明,如 val name = "Lee" 而不需要 String name = "Lee";

8. when 取代 switch

🔹 Java switch-case 语法繁琐

🔹 Kotlin when 语法更简洁,when 更直观,支持范围判断表达式返回值

9. 类默认 final

  • Java 类默认是 open 的,可以继承,除非加 final

  • Kotlin 类默认 final,必须显式 open 允许继承

10.总结对比表

特性JavaKotlin
语法冗长简洁
Null 安全可能导致 NPE避免 NPE
数据类需要手写 getter/setterdata class 自动生成
扩展函数需要工具类直接扩展
高阶函数需要匿名类直接支持 Lambda
并发线程(较重)协程(轻量)
类型转换需手动转换自动 Smart Cast
switch 语法switch-casewhen
类默认行为默认可继承默认 final

其他的:

11. Kotlin的懒加载

在 Kotlin 中,懒加载(Lazy Initialization)主要有两种方式:

  1. lazy适用于 val 只读变量
  2. lateinit适用于 var 可变变量

Kotlin by lazy和lateinit的使用及区别_kotlin by lazy 与lateinit-CSDN博客

12.单例模式

传统的懒汉式(lazy + @Volatile + synchronized)

class Singleton private constructor() {companion object {@Volatileprivate var instance: Singleton? = nullfun getInstance(): Singleton {return instance ?: synchronized(this) {//第一次空检查instance ?: Singleton().also { instance = it }//第二次空检查}}}
}

这里的双重检查:

  1. 第一次检查 (instance ?:)

    • 避免不必要的同步开销。
    • 如果已经初始化,直接返回,避免进入 synchronized 代码块,提高性能。
  2. 同步代码块内部的第二次检查 (instance ?:)

    • 由于多个线程可能同时通过第一次检查进入 synchronized,所以 需要再次检查 instance 是否为 null,防止重复创建实例。

为什么要用 @Volatile

  • @Volatile 防止指令重排序(保证可见性)。

  • 如果不加 @Volatile,可能会发生 部分初始化(对象创建未完成,别的线程就拿到不完整的实例)。

  • 避免可能的 NullPointerException(NPE)。

lazy懒加载

class Singleton private constructor() {companion object {val instance: Singleton by lazy { Singleton() }}
}
  • 线程安全(lazy 默认是 LazyThreadSafetyMode.SYNCHRONIZED)。
  • 更简洁,不需要 synchronized@Volatile

最简单的单例

object Singleton {fun doSomething() {println("Hello from Singleton!")}
}
  • 线程安全,在 Kotlin 中,object 关键字天然是线程安全的,因为它的初始化由 JVM 类加载机制(Class Loading Mechanism) 保证,由于 JVM 类加载过程是线程安全的,所以 object 也是线程安全的!

  • 写法简单

  • 类加载时就初始化(饿汉式)

🔹 双重检查锁的 Java 代码

public class Singleton {private static volatile Singleton instance;private Singleton() {}public static Singleton getInstance() {if (instance == null) {  // 第一次检查(避免不必要的同步)synchronized (Singleton.class) {  // 线程同步if (instance == null) {  // 第二次检查(确保只创建一次)instance = new Singleton();}}}return instance;}
}

13.kotlin的内联函数

  • Kotlin 允许使用 inline 关键字,将高阶函数的 函数调用 直接替换为 函数体代码,从而减少不必要的对象创建和额外的函数调用开销。
  • 主要作用:
    1. 避免高阶函数带来的 Lambda 表达式性能损耗。
    2. 允许 crossinlinenoinline 修饰参数,进一步控制内联行为。
inline fun execute(action: () -> Unit) {println("Before action")action()println("After action")
}fun main() {execute { println("Executing task...") }
}

等效于:

fun main() {println("Before action")println("Executing task...")println("After action")
}

execute() 方法的 Lambda 参数直接被展开,避免了额外的函数调用。

优点

  • 减少 Lambda 运行时开销(避免创建 Function 对象)
  • 提升代码执行效率
  • 支持非局部返回(可以使用 return 直接从调用函数返回)

注意事项

  • 避免过度使用,内联会导致代码膨胀,影响 APK 体积和方法数量。
  • 仅适用于小型 Lambda,对于大函数,内联会增加代码量而不是优化性能。

来个具体点的例子:

kotlin:

inline fun measureExecutionTime(crossinline beforeStart: () -> Unit,block: () -> Unit,crossinline afterEnd: () -> Unit
) {beforeStart()val startTime = System.currentTimeMillis()block()val endTime = System.currentTimeMillis()afterEnd()println("Execution Time: ${endTime - startTime} ms")
}fun main() {measureExecutionTime(beforeStart = { println("Starting execution...") },block = { Thread.sleep(500)  // 模拟耗时操作println("Executing task...")},afterEnd = { println("Execution completed.") })
}

java代码:

 

public class Main {public static void measureExecutionTime(Runnable beforeStart, Runnable block, Runnable afterEnd) {beforeStart.run();long startTime = System.currentTimeMillis();block.run();long endTime = System.currentTimeMillis();afterEnd.run();System.out.println("Execution Time: " + (endTime - startTime) + " ms");}public static void main(String[] args) {measureExecutionTime(() -> System.out.println("Starting execution..."),() -> {try {Thread.sleep(500); // 模拟耗时操作} catch (InterruptedException e) {e.printStackTrace();}System.out.println("Executing task...");},() -> System.out.println("Execution completed."));}
}

Java 代码中:

  • Java 需要为 Runnable 创建匿名内部类或 Lambda 对象,每次调用时都需要分配额外的内存。

  • Runnable 作为参数传递,执行 block.run(),涉及额外的函数调用

  • Java 的 Lambda 不能访问 return 语句,只能使用 return 结束 Runnable 的逻辑。

Kotlin 内联:

  • inline 使得 measureExecutionTime 不会创建任何 Lambda 对象,它的调用会被直接替换为代码块,消除对象分配和函数调用的开销。

  • 直接展开代码,避免额外的函数调用。

  • 支持非局部返回(普通 Lambda 不能直接 return 退出 measureExecutionTime,但 inline 可以)。

14. Any vs Object

在 Kotlin 中,Any 是所有非 null 类型的超类,而在 Java 中,Object 是所有类的超类。但它们并不完全等价,主要区别如下:

1. Any 不包含 null,而 Object 可以

  • 在 Kotlin 中,Any 不能存储 null,如果需要允许 null,必须使用 Any?

  • Java 的 Object 默认可以存储 null

2. Any 只有 equalshashCodetoString

  • Kotlin 的 Any 只包含 equals()hashCode()toString(),没有 wait()notify() 这些 Object 的方法。

  • Java 的 Object 还提供了 wait()notify()clone()线程相关的方法。

3. Any 不能直接用于 Java 互操作

在 Java 方法中,如果你需要一个 Object,Kotlin Any 不能直接替代它,必须显式转换为 Object

5. AnyUnitNothing 的超类,而 Object 不是

在 Kotlin,Unit(等价于 Java void)和 Nothing 也是 Any 的子类

15.智能类型转换(Smart Casts)

概念

  • Kotlin 编译器可以自动推断类型,避免手动类型转换 (cast),提升代码可读性和安全性。
  • 只要 Kotlin 确定变量不会改变类型,就可以自动转换。

java代码:

void printLength(Object obj) {if (obj instanceof String) {String str = (String) obj; // 需要手动转换System.out.println(str.length());}
}

kotlin代码:

fun printLength(obj: Any) {if (obj is String) { // 这里 obj 自动转换为 Stringprintln(obj.length) // 直接使用,不需要手动转换}
}

强制转换(asas?),其中as? 安全转换,避免 ClassCastException

val obj: Any = 123
val str: String? = obj as? String // 转换失败返回 null
println(str) // null


http://www.mrgr.cn/news/94257.html

相关文章:

  • 达梦主备集群部署
  • 阿里云操作系统控制台评测:国产AI+运维 一站式运维管理平台
  • ROS实践(四)机器人SLAM建图(gmapping)
  • 推理框架SGLang安装与调试
  • LVS + Keepalived 高可用集群
  • 《YOLOE: Real-Time Seeing Anything》论文速览翻译,支持文本提示,视觉提示等开放世界检测算法!
  • Java常见的并发设计模式
  • maven wrapper的使用
  • 爬虫中一些有用的用法
  • Qt:绘图API
  • 【Pytorch Transformers Fine-tune】使用BERT进行情感分类任务微调
  • Selenium 自动化测试学习总结
  • 本地Git仓库搭建(DevStar)与Git基本命令
  • MySQL的安装与建表
  • PySide(PyQT)的mouseMoveEvent()和hoverMoveEvent()的区别
  • java中小型公司面试预习资料(四):微服务架构
  • Unity 封装一个依赖于MonoBehaviour的计时器(上) 基本功能
  • Visual Studio 安装及使用教程(Windows)【安装】
  • JavaScript_Day2
  • 江科大51单片机笔记【16】AD/DA(上)