DexGuard 和矢量绘图

Posted

技术标签:

【中文标题】DexGuard 和矢量绘图【英文标题】:DexGuard and Vector Drawables 【发布时间】:2016-12-24 12:12:03 【问题描述】:

我正在使用 DexGuard 编译我的应用程序。我的应用程序使用矢量绘图。我的所有设置都是正确的,并且在调试版本和使用 Proguard 发布版本时都能正常工作。它只在使用 DexGuard 时在发布版本中崩溃。所以我知道这是一个 DexGuard 问题。 我正在使用 DexGuard v7.0,由于许可证限制,我无法更新。 我已经添加了 -keepresourcefiles "res/drawable/**" -keep class android.support.v7.** *; 到我的 dexguard 文件中用于测试目的和更清晰的堆栈跟踪,所以我们知道这不是问题。 它因以下堆栈跟踪而崩溃;

 java.lang.RuntimeException: Unable to start activity ComponentInfocom.lionscribe.myapp/com.lionscribe.elist.main.MainActivity: android.content.res.Resources$NotFoundException: File res/drawable/abc_ic_ab_back_material.xml from drawable resource ID #0x7f020018
       at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2413)
       at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2471)
       at android.app.ActivityThread.access$900(ActivityThread.java:175)
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1308)
       at android.os.Handler.dispatchMessage(Handler.java:102)
       at android.os.Looper.loop(Looper.java:146)
       at android.app.ActivityThread.main(ActivityThread.java:5602)
       at java.lang.reflect.Method.invokeNative(Native Method)
       at java.lang.reflect.Method.invoke(Method.java:515)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1283)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1099)
       at dalvik.system.NativeStart.main(Native Method)
    Caused by: android.content.res.Resources$NotFoundException: File res/drawable/abc_ic_ab_back_material.xml from drawable resource ID #0x7f020018
       at android.content.res.Resources.loadDrawable(Resources.java:3440)
       at android.content.res.Resources.getDrawable(Resources.java:1917)
       at o.?.?(:354)
       at android.support.v7.widget.AppCompatDrawableManager.getDrawable(:193)
       at android.support.v7.widget.AppCompatDrawableManager.getDrawable(:181)
       at android.support.v7.widget.AppCompatDrawableManager.checkVectorDrawableSetup(:689)
       at android.support.v7.widget.AppCompatDrawableManager.getDrawable(:186)
       at android.support.v7.widget.TintTypedArray.getDrawableIfKnown(:77)
       at android.support.v7.app.AppCompatDelegateImplBase.<init>(:83)
       at android.support.v7.app.AppCompatDelegateImplV7.<init>(:146)
       at android.support.v7.app.AppCompatDelegateImplV11.<init>(:28)
       at android.support.v7.app.AppCompatDelegateImplV14.<init>(:41)
       at android.support.v7.app.AppCompatDelegate.create(:193)
       at android.support.v7.app.AppCompatDelegate.create(:173)
       at android.support.v7.app.AppCompatActivity.getDelegate(:511)
       at android.support.v7.app.AppCompatActivity.onCreate(:71)
       at o.hF.onCreate(:29)
       at com.lionscribe.elist.main.MainActivity.onCreate(:121)
       at android.app.Activity.performCreate(Activity.java:5451)
       at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1093)
       at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2377)
       at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2471) 
       at android.app.ActivityThread.access$900(ActivityThread.java:175) 
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1308) 
       at android.os.Handler.dispatchMessage(Handler.java:102) 
       at android.os.Looper.loop(Looper.java:146) 
       at android.app.ActivityThread.main(ActivityThread.java:5602) 
       at java.lang.reflect.Method.invokeNative(Native Method) 
       at java.lang.reflect.Method.invoke(Method.java:515) 
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1283) 
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1099) 
       at dalvik.system.NativeStart.main(Native Method) 
    Caused by: org.xmlpull.v1.XmlPullParserException: Binary XML file line #17: invalid drawable tag vector
       at android.graphics.drawable.Drawable.createFromXmlInner(Drawable.java:986)
       at android.graphics.drawable.Drawable.createFromXml(Drawable.java:930)
       at android.content.res.Resources.loadDrawable(Resources.java:3436)
       at android.content.res.Resources.getDrawable(Resources.java:1917) 
       at o.?.?(:354) 
       at android.support.v7.widget.AppCompatDrawableManager.getDrawable(:193) 
       at android.support.v7.widget.AppCompatDrawableManager.getDrawable(:181) 
       at android.support.v7.widget.AppCompatDrawableManager.checkVectorDrawableSetup(:689) 
       at android.support.v7.widget.AppCompatDrawableManager.getDrawable(:186) 
       at android.support.v7.widget.TintTypedArray.getDrawableIfKnown(:77) 
       at android.support.v7.app.AppCompatDelegateImplBase.<init>(:83) 
       at android.support.v7.app.AppCompatDelegateImplV7.<init>(:146) 
       at android.support.v7.app.AppCompatDelegateImplV11.<init>(:28) 
       at android.support.v7.app.AppCompatDelegateImplV14.<init>(:41) 
       at android.support.v7.app.AppCompatDelegate.create(:193) 
       at android.support.v7.app.AppCompatDelegate.create(:173) 
       at android.support.v7.app.AppCompatActivity.getDelegate(:511) 
       at android.support.v7.app.AppCompatActivity.onCreate(:71) 
       at o.hF.onCreate(:29) 
       at com.lionscribe.elist.main.MainActivity.onCreate(:121) 
       at android.app.Activity.performCreate(Activity.java:5451) 
       at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1093) 
       at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2377) 
       at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2471) 
       at android.app.ActivityThread.access$900(ActivityThread.java:175) 
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1308) 
       at android.os.Handler.dispatchMessage(Handler.java:102) 
       at android.os.Looper.loop(Looper.java:146) 
       at android.app.ActivityThread.main(ActivityThread.java:5602) 
       at java.lang.reflect.Method.invokeNative(Native Method) 
       at java.lang.reflect.Method.invoke(Method.java:515) 
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1283) 
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1099) 
       at dalvik.system.NativeStart.main(Native Method) 

按照代码,我发现在使用 DexGuard 时,android.support.v7.widget.AppCompatDrawableManager 中的方法 loadDrawableFromDelegates(从 getDrawable 调用)返回 null。因此getDrawable 继续并调用ContextCompat.getDrawable,它不支持前棒棒糖设备中的Vector,它返回null 并带有invalid drawable tag vector 日志,因此导致ResourcesNotFoundException。 使用调试版本时,loadDrawableFromDelegates 确实返回了一个 Drawable,因此 ContextCompat.getDrawable 不会被调用,并且一切正常。 有谁知道如何避免这种情况。我用谷歌搜索过,还有其他人也有类似的问题,但主要是 appcompat 矢量设置问题,而不是 DexGuard。

【问题讨论】:

【参考方案1】:

经过 12 个小时的工作,我终于弄明白了,它只是 DexGuard 配置中的 1 行代码。您不需要我上面提到的任何设置。你只需要

-keepresourcexmlattributenames vector/**

原因是 AppCompat 库不查找向量的属性,例如通过 id 查找 viewportWidth,但它使用实际的属性名称。默认情况下,Dexguard 会删除名称,因此它从未找到该值。通过添加上述行,您告诉 DexGuard 保留所有矢量文件的属性名称。 我希望这可以减轻其他人的痛苦。

【讨论】:

您可以使用更新的 DexGuard 版本来省去麻烦,该版本已经通过其默认配置支持矢量可绘制对象。 我已经提到过,O 由于许可证限制无法升级。我相信有很多人处于相同的情况,因为 DexGuard 已经改变了他们的许可模式。 好吧,“许可限制”有点误导。 DexGuard 已将其许可模式更改为按年订阅,对于某些开发人员来说可能有点太贵了,但本身并没有限制。 嗯,将费用从每位用户无限制的 500 欧元转换为每个应用程序每年约 5000 欧元,对大多数人来说是一个大问题,并且是一个不允许升级的限制!

以上是关于DexGuard 和矢量绘图的主要内容,如果未能解决你的问题,请参考以下文章

Sketch Mac版是一个创新和新的眼光看待矢量绘图软件

CDR是什么?CorelDRAW矢量绘图

缩放会导致像素化矢量绘图

Egret 矢量绘图遮罩碰撞检测

矢量图形(高效绘图 13.2)

优秀的矢量绘图和图形设计工具Amadine for Mac 1.0