最近碰见了一个需求,客户提供的代码中大量使用了SDK中标注为@hide的类和方法。
然后代码就一路报错。并且提示找不到该方法和工具。 但整个项目能够在系统源码下进行生成正常的APK并运行。
那么我们如果想再Android Studio中进行编译该怎么办?想方便灵活的进行应用层数据的编辑和修改在源码下实在不方便。
那么你就可以参考使用Layoutlib.jar 包来实现。
第一步
选择有利的Layoutlib.jar
api 25 以上的layoutlib不行。没有源码了。
将layoutlib.jar 复制到 你的app module 下面 任意新建一个文件夹;注意,请不要放到libs文件夹下,否则编译时会出现下面的错误
AGPBI: {"kind":"error","text":"Cannot fit requested classes in a single dex file. Try supplying a main-dex list.\r\n# methods: 72366 \u003e 65536","sources":[{}],"tool":"D8"}
FAILED
FAILURE: Build failed with an exception.
* What went wrong:
Execution failed for task ':app:transformDexArchiveWithExternalLibsDexMergerForDebug'.
> com.android.builder.dexing.DexArchiveMergerException: Error while merging dex archives: D:\androidProject\WifiDisplay-Miracast\WifiDisplay\app\build\intermediates\transforms\dexBuilder\debug\2.jar, D:\androidProject\WifiDisplay-Miracast\WifiDisplay\app\build\intermediates\transforms\dexBuilder\debug\1.jar, D:\androidProject\WifiDisplay-Miracast\WifiDisplay\app\build\intermediates\transforms\dexBuilder\debug\3.jar
The number of method references in a .dex file cannot exceed 64K.
Learn how to resolve this issue at https://developer.android.com/tools/building/multidex.html
* Try:
Run with --stacktrace option to get the stack trace. Run with --info or --debug option to get more log output. Run with --scan to get full insights.
* Get more help at https://help.gradle.org
BUILD FAILED in 57s
21 actionable tasks: 3 executed, 18 up-to-date
提示方法量超过了65536行。主要原因是apk生成的时候 将layoutlib.jar 库打包进了apk之中了。
我们使用layoutlib只是想避免开发编译时的错误,而不是为了将layoutlib打包进apk之中。因为apk生成成功后运行时会自动调用系统里面的framework 库,相关的类和方法都是存在的。
存储目录如图:
第二步:
添加成功后,需要加载jar
在app下面的 build.gradle中添加下面的代码:
dependencies {
implementation fileTree(include: ['*.jar'], dir: 'libs')
compileOnly files('syslibs/layoutlib.jar')
}
//主要修改iml 配置,将SDK的顺序降低到layoutlib.jar 之下。
preBuild {
doLast {
def imlFile = file(project.name + ".iml")
println('Change ' + project.name + '.iml order')
try {
def parsedXml = (new XmlParser()).parse(imlFile)
def jdkNode = parsedXml.component[1].orderEntry.find { it.'@type' == 'jdk' }
parsedXml.component[1].remove(jdkNode)
def sdkString = "Android API " + android.compileSdkVersion.substring("android-".length()) + " Platform"
new groovy.util.Node(parsedXml.component[1], 'orderEntry', ['type': 'jdk', 'jdkName': sdkString, 'jdkType': 'Android SDK'])
groovy.xml.XmlUtil.serialize(parsedXml, new FileOutputStream(imlFile))
} catch (FileNotFoundException e) {
// nop, iml not found
}
}
}
Gradle之后 打开app.iml 类 就能看到下面图片的效果,刚才加载的jar包会先SDK的jar被引用。
到这一步为止,项目就不会有错误了。
刚才如果将layoutlib.jar 放到libs目录下, 那么就会在生成的时候 将其打包到apk中,而出现64K的错误。
64k 的错误也可以采用分包的模式减小大小实现,但是就没有意义了。所以建议别放到libs文件下。
如果整个项目不会使用外部的jar包。那可以放到libs目录下, 但是需要注释掉下面的代码
dependencies {
// implementation fileTree(include: ['*.jar'], dir: 'libs') 注释掉这一行
compileOnly files('libs/layoutlib.jar')
}
第三步:
最后 编译打包成apk时又会提示类不存储,方法不存在。那是因为在编译生成数据的时候检查类不存在。
我们需要在最外面的build.gradle中进行修改
allprojects {
repositories {
google()
jcenter()
}
//添加这一行 其中app 根据你的module的名称进行替换, 后面的路径文件名都可以根据实际情况进行替换,但是开头的-Xbootclasspath/p: 这个是关键字
gradle.projectsEvaluated {
tasks.withType(JavaCompile) {
options.compilerArgs.add('-Xbootclasspath/p:app\\syslibs\\layoutlib.jar')
}
}
}
到现在 问题就能解决了。
第一次碰见这种的需求进行调整。感觉走的是邪路呢