位置: 编程技术 - 正文
推荐整理分享ApkTool原码修改记录(apktool修改包名),希望有所帮助,仅作参考,欢迎阅读内容。
文章相关热门搜索词:apk 修改,apktool修改版本号,apktool修改软件安卓版本,apktool修改包名,apktool修改包名,apktool修改软件安卓版本,apktool修改软件安卓版本,apktool修改版本号,内容如对您有帮助,希望把文章链接给更多的朋友!
转载请注明:本文源自 in thread "main" java.lang.ClassCastException: brut.androlib.res.data.value.ResStringValue cannot be cast to brut.androlib.res.dat at brut.androlib.res.decoder.ResAttrDecoder.decode(ResAttrDecoder.java:) at brut.androlib.res.decoder.AXmlResourceParser.getAttributeValue(AXmlResourceParser.java:) at org.xmlpull.v1.wrapper.classic.XmlPullParserDelegate.getAttributeValue(XmlPullParserDelegate.java:) at org.xmlpull.v1.wrapper.classic.StaticXmlSerializerWrapper.writeStartTag(StaticXmlSerializerWrapper.java:) at org.xmlpull.v1.wrapper.classic.StaticXmlSerializerWrapper.event(StaticXmlSerializerWrapper.java:) at brut.androlib.res.decoder.XmlPullStreamDecoder$1.event(XmlPullStreamDecoder.java:) at brut.androlib.res.decoder.XmlPullStreamDecoder.decode(XmlPullStreamDecoder.java:) at brut.androlib.res.decoder.ResStreamDecoderContainer.decode(ResStreamDecoderContainer.java:) at brut.androlib.res.decoder.ResFileDecoder.decode(ResFileDecoder.java:) at brut.androlib.res.decoder.ResFileDecoder.decode(ResFileDecoder.java:) at brut.androlib.res.AndrolibResources.decode(AndrolibResources.java:) at brut.androlib.Androlib.decodeResourcesFull(Androlib.java:) at brut.androlib.ApkDecoder.decode(ApkDecoder.java:) at brut.apktool.Main.cmdDecode(Main.java:) at brut.apktool.Main.main(Main.java:)
网上搜时,要用最新的apktool.jar,于是,下载了apktool_2.0.0rc3.jar,apktool_2.0.0rc2.jar,apktool_2.0.0rc4.jar,估计4是网上大神自己发布的,但没有一个能正确反编译xml,都报错,有人给出答案:加-r参数:pktool d -r xx.apk xx,其实这样做,xml是解压出来的,根本没解析。折腾了好久,没搞定,一怒之下,找了apktool源码,但这个源码是在Google服务器上的一个开源项目,国内的环境,你懂的!没办法,只能下载一个,忘了这个下载地址在哪,真的对不起,但我把修改过的源码放到了开源中国git服务器上,git地址: brut.androlib.res.decoder.ResAttrDecoder.decode(ResAttrDecoder.java:)这个地方,发现原来是类型强转不对,接下问题就好解了
问题出在这:
ResAttr attr = (ResAttr) getCurrentPackage().getResTable() .getResSpec(attrResId).getDefaultResource().getValue(); decoded=attr.convertToResXmlFormat(resValue);
看了下,getCurrentPackage().getResTable().getResSpec(attrResId).getDefaultResource().getValue();报错时返回是:ResStringValue类型的,ResStringValue与ResAttr其实不是父子关系,而是兄弟关系,所以强转肯定会报错的,解决方法:
ResValue attr = (ResValue) getCurrentPackage().getResTable() .getResSpec(attrResId).getDefaultResource().getValue(); if (attr instanceof ResStringValue) { decoded=((ResStringValue)attr).encodeAsResXmlAttr();// System.out.println("decoded="decoded); } else if (attr instanceof ResAttr) { decoded = ((ResAttr) attr).convertToResXmlFormat(resValue); }
判断下返回的类型,如果是ResStringValue,就按ResStringValue类型处理,如果是ResAttr,就按ResAttr类型处理 ,再反编译下,竟然正常了!有点小激动!
反编译完后,再回编译,又陷入了坑中……但这反编译是没问题的,用其他版本apktool.jar可正常回编译,但这个源码处,怎么也不能正常回编!
苦思中……
很久……
没办法,再来,把回编译流程跟踪了一遍,再看下其他版本的apktool.jar代码,(通过gd-gui.exe)再把相关的函数对比了下,竟然没发现不同!!!
绝望了……
只好从报错处开始研究,发现是这个变量引起的:mAaptPath
mAaptPath是aapt.exe文件路径,调试时,发现如果加入-a ./aapt.exe,就可以正常回编,知道问题所在了,如果没指定-a参数,程序就会自动去找aapt.exe,如果没找到就会报错,然后跟踪了找aapt.exe文件的函数:
在Androlib.java中
mAndRes.aaptPackage(apkFile, new File(appDir, "AndroidManifest.xml"), new File(appDir, "res"), ninePatch, null, parseUsesFramework(usesFramework), flags, mAaptPath);
往下走:AndrolibResources.java中的aaptPackage中:
这是我修改过的的代码,加入了打印
if (!aaptPath.isEmpty()) { File aaptFile = new File(aaptPath); if (aaptFile.canRead() && aaptFile.exists()) { aaptFile.setExecutable(true); cmd.add(aaptFile.getPath()); LogDebug.i("aaptPackage 1"); customAapt = true; if (flags.get("verbose")) { LOGGER.info(aaptFile.getPath() " being used as aapt location."); } } else { LOGGER.warning("aapt location could not be found. Defaulting back to default"); try { LogDebug.i("aaptPackage 2"); cmd.add(getAaptBinaryFile().getAbsolutePath()); } catch (BrutException ignored) { LogDebug.i("aaptPackage 3"); cmd.add("aapt"); } } } else { try { File aaptFile = new File(getAaptBinaryFile().getAbsolutePath()); if (aaptFile.canRead() && aaptFile.exists()) { aaptFile.setExecutable(true); cmd.add(aaptFile.getPath()); LogDebug.i("aaptPackage "); customAapt = true; if (flags.get("verbose")) { LOGGER.info(aaptFile.getPath() " being used as aapt location."); } } else { cmd.add(getAaptBinaryFile().getAbsolutePath()); LogDebug.i("aaptPackage 4" getAaptBinaryFile().getAbsolutePath()); } } catch (BrutException ignored) { LogDebug.i("aaptPackage 5"); cmd.add("aapt"); } }
关键是:getAaptBinaryFile().getAbsolutePath()这个方法,出问题了,有问题就得改:
在getAaptBinaryFile函数中,有三个分支:
try { if (OSDetection.isMacOSX()) { mAaptBinary = Jar .getResourceAsFile("/prebuilt/aapt/macosx/aapt"); } else if (OSDetection.isUnix()) { mAaptBinary = Jar .getResourceAsFile("/prebuilt/aapt/linux/aapt"); } else if (OSDetection.isWindows()) { // mAaptBinary = // Jar.getResourceAsFile("/prebuilt/aapt/windows/aapt.exe"); mAaptBinary = new File(getAaptPath()); } else { LOGGER.warning("Unknown Operating System: " OSDetection.returnOS()); return null; } } catch (BrutException ex) { throw new AndrolibException(ex); }
(以上代码注释部分为源码)
看方法名也知道,其他两个系统:Mac和Unix我就没去研究了,只是在Windows系统下调试,修改:mAaptBinary = new File(getAaptPath());
其中:getAaptPath()方法就是获取aapt.exe文件路径,实现原理:用java执行cmd命令:echo %PATH%
把得到的字符串(系统变量中的path变量,用“;”分隔)分割成字符串数组,对数组中的每个元素进行扫描(每个元素其实是目录地址),直到找到aapt.exe,就返回该文件地址,具体实现请参照代码中的写法!
纯手打,望尊重程序员的劳动成果!
android图片透明度跟缩放大小动画事件 概序:动画事件写在xml中,然后用AnimationUtils去加载动画事件,再监听动画结束事件,隐藏imageview。1.player_double_click_animation.xml动画文件?xmlversion=1.0encoding=ut
[android] android开发中常用的Eclipse快捷键 Eclipse快捷键-方便查找,呵呵,记性不好行注释/销注释Ctrl/块注释/销注释/XML注释CtrlShift/CtrlShift查找查找替换CtrlHCtrlF查找下一个/往回找CtrlKCtrlShiftK跳到
【面试题】JAVA 一个线程依赖另外一个线程的结果 publicclassMain{publicstaticclassMyCallableimplementsCallableInteger{publicIntegercall()throwsException{return1;}}publicstaticvoidmain(String[]args){MyCallablecallable=newMyCallable();FutureTaskIntege
标签: apktool修改包名
本文链接地址:https://www.jiuchutong.com/biancheng/376646.html 转载请保留说明!上一篇:Android网络请求开源框架Volley快速使用指南(2)——异步获取图片(android网络请求时怎么取消)
下一篇:android图片透明度跟缩放大小动画事件(android图片透明度怎么设置)
友情链接: 武汉网站建设