如何使用 PathClassLoader 替换已弃用的 DexFile API?

Car*_*son 6 android jvm classloader dalvik dex

我有一个类,我在其中进行了一些运行时注释扫描,但它使用了已弃用的 DexFile API,这会导致 LogCat 中出现警告:

  • W/zygote64: Opening an oat file without a class loader. Are you using the deprecated DexFile APIs?.

我想摆脱此消息并使用适当的 API。文档建议PathClassLoader,但我不明白它DexFile在功能上的等效性。我可以将 aPathClassLoaderDexFile实例结合使用,虽然它确实有效,但它会给我更多警告并且需要更长的时间来扫描。为了清楚起见,我已经包含了我在下面写的注释扫描器。如果有人可以建议如何摆脱这些警告消息和 的替代方案DexFile,因此在删除后我不会受到损坏的功能的影响,我会非常感激。

class AnnotationScanner {
    companion object {
        fun classesWithAnnotation(
            context: Context,
            annotationClass: Class<out Annotation>,
            packageName: String? = null
        ): Set<Class<*>> {

            return Pair(context.packageCodePath, context.classLoader)
                .letAllNotNull { packageCodePath, classLoader ->
                    Pair(DexFile(packageCodePath), classLoader)
                }
                ?.letAllNotNull { dexFile, classLoader ->
                    dexFile
                        .entries()
                        ?.toList()
                        ?.filter { entry ->
                            filterByPackageName(packageName, entry)
                        }
                        ?.map {
                            dexFile.loadClass(it, classLoader)
                        }
                        ?.filter { aClass ->
                            filterByAnnotation(aClass, annotationClass)
                        }
                        ?.toSet()
                } ?: emptySet<Class<*>>().wlog { "No ${annotationClass.simpleName} annotated classes found" }
        }

        private fun filterByAnnotation(aClass: Class<*>?, annotationClass: Class<out Annotation>): Boolean {
            return aClass
                ?.isAnnotationPresent(annotationClass)
                ?.also {
                    it.ifTrue {
                        Timber.w("Found ${annotationClass.simpleName} on $aClass")
                    }
                }
                ?: false
        }

        private fun filterByPackageName(packageName: String?, entry: String) =
            packageName?.let { entry.toLowerCase().startsWith(it.toLowerCase()) } ?: true
    }
}
Run Code Online (Sandbox Code Playgroud)