我正在尝试使用Kotlin类重装,但最近我遇到了一些与此相关的问题:
package com.aurieh.reloading
fun doSomething(): String { // a function that does not belong to the class,
// so it gets compiled as FileName$doSomething$...
}
class FileName {
// do things with doSomething
}
如果我重新加载这个类(使用ImplClassLoader.defineClass
和ByteArray
),并尝试调用内部调用doSomething的方法,我会收到类似于的错误:
java.lang.IllegalAccessError: tried to access method com.aurieh.reloading.FileName.doSomething$default()Ljava/lang/String; from class com.aurieh.ares.reloading.FileName`
我会将此解释为重新加载的类没有附加doSomething .. 所以我的问题是,我将如何解决这个错误?通过某种方式将doSomething附加到重新加载类加载器?
供参考,我的班级重装代码:
class Reloader : ClassLoader() {
fun load(name: String, bytes: ByteArray, offset: Int, len: Int): Class<*> {
return defineClass("com.aurieh.reloading.$name", bytes, offset, len)
}
}
加载:
val bytes = File("../classes/path/to/class/FileName.class").readBytes()
Reloader().load("FileName", bytes, 0, bytes.size).newInstance()
答案 0 :(得分:5)
基本上,顶级函数不会编译到文件中定义的任何类中。而是为顶级成员创建一个单独的类:FileNameKt
(如果文件名为FileName.kt
)。
因此,要正确加载类(即没有任何不满意的链接),您必须首先加载FileNameKt
类:
val bytes1 = File("../classes/path/to/class/FileNameKt.class").readBytes()
val bytes2 = File("../classes/path/to/class/FileName.class").readBytes()
val reloader = Reloader()
reloader.load("FileNameKt", bytes1, 0, bytes1.size)
reloader.load("FileName", bytes2, 0, bytes2.size).newInstance()