一、Apktool 的使用

相信对于熟悉 Android 开发的人来说,Apktool 应该是十分熟悉的。特别是对于游戏发行商来说,很多情况下是直接拿 APK 通过 Apktool 反编译,替换参数和资源文件,再重编译,重签达到渠道分发的目的。

1、反编译 APK

1
apktool d -f --only-main-classes /aaa/bbb/ccc.apk -o /aaa/bbb/ooo

2、重编译 APK

1
apktool b -f /aaa/bbb/ooo

二、APKTOOL_DUMMY_* 问题

有时候我们使用该工具反编译再重编译,发现重编译失败,报错类似这种:

翻阅了 apktool 的 Issues,找到了以下几篇相关的:

根据以上得出结论,APKTOOL_DUMMY_** 是工具在反编译时对“稀缺资源”进行补齐,至于这里的“稀缺资源”具体是指什么,可能得研究下源码才行。

使用 v2.5.0 版本,反编之后,工具会自动加上一些类似这样的:

1
2
3
4
/res/values/colors.xml:    <color name="APKTOOL_DUMMY_2d" />
/res/values/colors.xml: <color name="APKTOOL_DUMMY_2e" />
/res/values/colors.xml: <color name="APKTOOL_DUMMY_6f" />
...

这里重编译是失败的。

而使用 v2.6.1 版本,反编之后,工具会自动加上一些类似这样的:

1
2
3
4
/res/values/colors.xml:    <item type="color" name="APKTOOL_DUMMY_2d" />
/res/values/colors.xml: <item type="color" name="APKTOOL_DUMMY_2e" />
/res/values/colors.xml: <item type="color" name="APKTOOL_DUMMY_6f" />
...

这里重编译则是成功的。应该是补齐的代码的写法修改,使得编译通过。

而在 v2.7.0 版本以及之后,甚至都不会有 APKTOOL_DUMMY_** ,说明 APKTOOL_DUMMY_** 的引入不仅没用,还可能导致重编译失败。
因此如果遇到上述的问题,使用最新的工具应该可以解决。如果还是想坚持使用旧版本的工具,这里提供一个思路:
批量读取 .xml 文件(AndroidManifest.xml除外)每一行,如果出现 APKTOOL_DUMMY_,则移除该行并保存,当所有 APKTOOL_DUMMY_ 被移除后,重新编译是正常的,且 APK 能够正常运行。脚本代码如下(Python3):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#!/usr/bin/env python

import os
import argparse

def remove_dummy_lines(directory):
# 遍历指定目录及其子目录下的所有文件
for root, dirs, files in os.walk(directory):
for file in files:
# 忽略 AndroidManifest.xml
if file in ['AndroidManifest.xml']:
continue
if file.endswith('.xml'): # 确保处理的是 XML 文件
file_path = os.path.join(root, file)
with open(file_path, 'r', encoding='utf-8') as f:
lines = f.readlines() # 读取所有行

# 创建一个新的列表来存储修改后的行
new_lines = []
for line in lines:
if 'APKTOOL_DUMMY_' not in line:
new_lines.append(line)

# 如果有行被删除,重新写入文件
if len(new_lines) < len(lines):
with open(file_path, 'w', encoding='utf-8') as f:
f.writelines(new_lines)
print(f"Updated file: {file_path}")

if __name__ == "__main__":

parser = argparse.ArgumentParser("remove apktool placeholder")
parser.add_argument("-decompile_path", dest="decompile_path", help="输入apk反编译的文件夹")
args = parser.parse_args()
decompile_path = args.decompile_path

# 调用函数,传入要遍历的目录路径
remove_dummy_lines(decompile_path)

将代码保存到文件 removeApktoolPlaceholder.py,在终端调用方式:

1
python3 removeApktoolPlaceholder.py -decompile_path /your/path/

三、参考链接