- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我有一些代码来获取包中所有类的列表,看起来像这样:
try {
DexFile df = new DexFile(context.getPackageCodePath());
for (Enumeration<String> iter = df.entries(); iter.hasMoreElements();) {
String s = iter.nextElement();
}
} catch (IOException e) {
e.printStackTrace();
}
但是,自从我将 Android Studio 升级到 2.0 版后,这段代码就停止工作了。我发现罪魁祸首是 Instant Run。如果我调试应用程序,我可以看到在没有实例运行的情况下,DexFile 变量 df 包含一个类名列表(超过 4,000 个)。当 Instant Run 打开时,我只得到大约 30 个类名,而我要查找的类不存在。我感觉它与 multidex 有关,但我不确定 Instant Run 在幕后是如何工作的(我的应用程序不使用 multidex)。
有人知道如何在启用 Instant Run 的情况下获得这样的类列表吗?或者有人确切地知道为什么我会看到这种行为(理解它会很棒)吗?
最佳答案
我们可以在应用程序数据路径中处理即时运行构建的DEX文件。
public class MultiDexHelper {
private static final String EXTRACTED_NAME_EXT = ".classes";
private static final String EXTRACTED_SUFFIX = ".zip";
private static final String SECONDARY_FOLDER_NAME = "code_cache" + File.separator +
"secondary-dexes";
private static final String PREFS_FILE = "multidex.version";
private static final String KEY_DEX_NUMBER = "dex.number";
private static SharedPreferences getMultiDexPreferences(Context context) {
return context.getSharedPreferences(PREFS_FILE, Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB ?
Context.MODE_PRIVATE :
Context.MODE_PRIVATE | Context.MODE_MULTI_PROCESS);
}
/**
* get all the dex path
*
* @param context the application context
* @return all the dex path
* @throws PackageManager.NameNotFoundException
* @throws IOException
*/
public static List<String> getSourcePaths(Context context) throws PackageManager.NameNotFoundException, IOException {
ApplicationInfo applicationInfo = context.getPackageManager().getApplicationInfo(context.getPackageName(), 0);
File sourceApk = new File(applicationInfo.sourceDir);
File dexDir = new File(applicationInfo.dataDir, SECONDARY_FOLDER_NAME);
if (LogUtil.isDebugModeEnable()) {
LogUtil.d("MultiDexHelper",
"getSourcePaths sourceDir=" + applicationInfo.sourceDir + ", dataDir=" + applicationInfo.dataDir);
}
List<String> sourcePaths = new ArrayList<String>();
sourcePaths.add(applicationInfo.sourceDir); //add the default apk path
//the prefix of extracted file, ie: test.classes
String extractedFilePrefix = sourceApk.getName() + EXTRACTED_NAME_EXT;
//the total dex numbers
int totalDexNumber = getMultiDexPreferences(context).getInt(KEY_DEX_NUMBER, 1);
if (LogUtil.isDebugModeEnable()) {
LogUtil.d("MultiDexHelper", "getSourcePaths totalDexNumber=" + totalDexNumber);
}
for (int secondaryNumber = 2; secondaryNumber <= totalDexNumber; secondaryNumber++) {
//for each dex file, ie: test.classes2.zip, test.classes3.zip...
String fileName = extractedFilePrefix + secondaryNumber + EXTRACTED_SUFFIX;
File extractedFile = new File(dexDir, fileName);
if (extractedFile.isFile()) {
sourcePaths.add(extractedFile.getAbsolutePath());
//we ignore the verify zip part
} else {
throw new IOException("Missing extracted secondary dex file '" +
extractedFile.getPath() + "'");
}
}
try {
// handle dex files built by instant run
File instantRunFilePath = new File(applicationInfo.dataDir,
"files" + File.separator + "instant-run" + File.separator + "dex");
if (LogUtil.isDebugModeEnable()) {
LogUtil.d("MultiDexHelper", "getSourcePaths instantRunFile exists=" + instantRunFilePath.exists() + ", isDirectory="
+ instantRunFilePath.isDirectory() + ", getAbsolutePath=" + instantRunFilePath.getAbsolutePath());
}
if (instantRunFilePath.exists() && instantRunFilePath.isDirectory()) {
File[] sliceFiles = instantRunFilePath.listFiles();
for (File sliceFile : sliceFiles) {
if (null != sliceFile && sliceFile.exists() && sliceFile.isFile() && sliceFile.getName().endsWith(".dex")) {
sourcePaths.add(sliceFile.getAbsolutePath());
}
}
}
} catch (Throwable e) {
LogUtil.e("MultiDexHelper", "getSourcePaths parse instantRunFilePath exception", e);
}
return sourcePaths;
}
// /**
// * get all the classes name in "classes.dex", "classes2.dex", ....
// *
// * @param context the application context
// * @return all the classes name
// * @throws PackageManager.NameNotFoundException
// * @throws IOException
// */
// public static List<String> getAllClasses(Context context) throws PackageManager.NameNotFoundException, IOException {
// List<String> classNames = new ArrayList<String>();
// for (String path : getSourcePaths(context)) {
// try {
// DexFile dexfile = null;
// if (path.endsWith(EXTRACTED_SUFFIX)) {
// //NOT use new DexFile(path), because it will throw "permission error in /data/dalvik-cache"
// dexfile = DexFile.loadDex(path, path + ".tmp", 0);
// } else {
// dexfile = new DexFile(path);
// }
// Enumeration<String> dexEntries = dexfile.entries();
// while (dexEntries.hasMoreElements()) {
// classNames.add(dexEntries.nextElement());
// }
// } catch (IOException e) {
// throw new IOException("Error at loading dex file '" +
// path + "'");
// }
// }
// return classNames;
// }
/**
* scan parent class's sub classes
*
* @param context
* @param packageName
* @param parentClass
* @param <T>
* @return
*/
public static <T> Set<Class<? extends T>> scanClasses(Context context, String packageName, Class<T> parentClass) {
Set<Class<? extends T>> classes = new HashSet<Class<? extends T>>();
try {
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
for (String path : getSourcePaths(context)) {
if (LogUtil.isDebugModeEnable()) {
LogUtil.d("MultiDexHelper", "scanClasses path=" + path);
}
try {
DexFile dexfile = null;
if (path.endsWith(EXTRACTED_SUFFIX)) {
//NOT use new DexFile(path), because it will throw "permission error in /data/dalvik-cache"
dexfile = DexFile.loadDex(path, path + ".tmp", 0);
} else {
dexfile = new DexFile(path);
}
Enumeration<String> dexEntries = dexfile.entries();
while (dexEntries.hasMoreElements()) {
String className = dexEntries.nextElement();
if (LogUtil.isDebugModeEnable()) {
LogUtil.d("MultiDexHelper", "scanClasses className=" + className);
}
if (className.toLowerCase().startsWith(packageName.toLowerCase())) {
Class clazz = classLoader.loadClass(className);
if (LogUtil.isDebugModeEnable()) {
LogUtil.d("MultiDexHelper",
"scanClasses clazz=" + clazz + ", parentClass=" + parentClass + ", equals=" + clazz
.getSuperclass().equals(parentClass));
}
if (clazz.getSuperclass().equals(parentClass)) {
classes.add(clazz);
}
}
}
} catch (Throwable e) {
LogUtil.e("MultiDexHelper", "scanClasses Error at loading dex file '" +
path + "'", e);
}
}
} catch (Throwable e) {
LogUtil.e("MultiDexHelper", "scanClasses exception", e);
}
return classes;
}
关于Android Studio 2.0 Instant Run 导致 DexFile 无法加载所有类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36491692/
例如,如果一个 Instant 在其他两个 Instant 的开始和结束范围内,您想在 Kotlin 中检查,您可以简单地执行以下操作: import java.time.Instant val s
Google 即搜即得是一种新的搜索增强功能,可在您键入时显示结果。在 the information page for Google Instant ,它说“15 项新技术有助于 Google 即搜
我定义了以下模型 public class ItemDetail { private final String name; private final String id; p
游戏已启动,我收到了用户的状态,一切就绪。我正在尝试构建一个问答游戏。我正在从远程服务器获取所有信息,其中包括基于问题的图像 Assets 。我可以获取远程数据,但无法显示图像。看来 facebook
我正在尝试生成 java.time.Instant 的实例用于基于属性的测试,并使用Temporal.range几乎适用于所有 Temporal我尝试过的子类- 除了Instant ,这似乎没有提供有
时间戳将存储在数据库中。确定一个名为“b”的时间戳是否是另一个名为“a”的时间戳的后一天的解决方案是什么? 时区将作为参数提供。 例如,考虑: Instant a = Instant.ofEpochS
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 8 年前。 Improve
我想权衡我购买其中一个或两个订阅的决定。我想得到一些数据来支持我的决定。是否可以编写一个程序来比较库?可以从 netflix 和 amazon 获得 API 吗? 最佳答案 除了 scaba 的回答之
我有一个 Instant (org.joda.time.Instant) 的实例,我在一些 api 响应中得到它。我有另一个来自 (java.time.Instant) 的实例,这是我从其他调用中获得
我正在尝试将 org.thirden.bp.LocalDate 转换为 java.util.Date,并且收到问题标题中提到的错误。 我使用以下进行转换: Date.from(currentDate.
对于我的每个用户,我存储一个 tzid我将其转换为 DateTimeZone持有有关其本地时区的信息。 我想在本地时间每天早上 8 点向用户发送一封电子邮件;如果上午 8 点由于某种原因(例如夏令时类
本文将提供如何在 Java LocalDateTime 和 Instant 之间进行转换。 LocalDateTime 表示没有时区的日期时间,例如 2019-10-25T12:15:30,而 Ins
这个问题在这里已经有了答案: 10年前关闭。 Possible Duplicate: How does Google Instant work? 我们经常使用谷歌搜索任何内容,但是当我们在谷歌引擎的文
我有一个设计类似于 myspace/facebook 类型社区的社交网站,我在 LAMP 设置上使用 php/mysql。 我一直想拥有自己的即时通讯工具,可以在用户的 PC 上运行,类似于 AI
我正在为我的网站开发私有(private)消息传递功能, 有没有办法从服务器获取用户收到新消息的推送通知? 我知道的唯一技术是通过 ajax 不断轮询服务器以查看是否有新消息,并在需要时重新加载信使窗
我越来越了解NodaTime并且非常喜欢它。但是我还不太了解! 给定诸如“ 2014-04-08T09:30:18Z”的值,将这样的字符串解析为NodaTime Instant所需的步骤是什么? 谢谢
这个问题已经有答案了: How to unify date format using DateTimeFormatter (4 个回答) What is the Standard way to Par
我正在尝试根据 B.C.E. 创建一个 Instant。公历年。 这是我目前所拥有的: Instant.FromDateTimeOffset(new DateTimeOffset(-1000, 10,
我已尝试获取特定时区的当前实例,但它没有按预期工作。关于如何执行此操作的任何想法? 这是我做的: Instant.now(Clock.system(ZoneId.of("America/Los_Ang
这个问题可能与另一个问题类似: HTML Comparing 2 Dates with Javascript 但我不想使用按钮,只需让它即时即可。 我有 2 个类型为 date 的输入,我想比较它们。
我是一名优秀的程序员,十分优秀!