inline 本质
Kotlin中的内联函数使用inline
关键字修饰,形如:
inline fun sum(x:Int, y:Int)
编译时,inline
会将内联函数中的代码直接复制到调用处
inline 使用场景
Kotlin 推荐在传入函数类型参数的函数上(lambda 参数)使用:
[inline] fun measureTime(action : ()-> Unit) {
val start = System.currentTimeMillis()
action()
val end = System.currentTimeMillis()
println("<<<< ${end - start}ms")
}
fun main() {
measureTime { println("Long Long ago") }
}
上述代码在编译后:
- 使用了 inline 关键词:measureTime 方法体被直接复制到 main 函数中
public static final void measureTime(@NotNull Function0 action) {
int $i$f$measureTime = 0;
Intrinsics.checkParameterIsNotNull(action, "action");
long start = System.currentTimeMillis();
action.invoke();
long end = System.currentTimeMillis();
String var6 = "<<<< " + (end - start) + "ms";
boolean var7 = false;
System.out.println(var6);
}
public static final void main() {
int $i$f$measureTime = false;
long start$iv = System.currentTimeMillis();
int var3 = false;
String var4 = "Long Long ago";
boolean var5 = false;
System.out.println(var4);
long end$iv = System.currentTimeMillis();
String var9 = "<<<< " + (end$iv - start$iv) + "ms";
boolean var8 = false;
System.out.println(var9);
}
- 未使用 inline 关键词:measureTime 调用时传入了一个额外对象
null.INSTANCE
public static final void measureTime(@NotNull Function0 action) {
Intrinsics.checkParameterIsNotNull(action, "action");
long start = System.currentTimeMillis();
action.invoke();
long end = System.currentTimeMillis();
String var5 = "<<<< " + (end - start) + "ms";
boolean var6 = false;
System.out.println(var5);
}
public static final void main() {
measureTime((Function0)null.INSTANCE);
}
在不加 inline 的情况下,measureTime 若频繁调用,则会产生很多不必要的额外INSTANCE。
inline 真正优势 拯救 lambda
Kotlin 内联函数的作用是消除 lambda 带来的额外开销
在传入函数类型的函数上修饰,避免调用方创建不必要的额外对象
inline 配合 reified 关键字,可使用泛型对象属性
// 不加 reified 关键字,报错‘cannot use T as reified parameter, use a class instead’
inline fun <T> create() {
val clazzT = T::class.java
}
// 正确写法
inline fun <reified T> create() {
val clazzT = T::class.java
}