在回答您的问题之前,这里简要介绍一下两种存储类型:
缓存
这是文件系统上特定于应用程序的目录。该目录旨在存储应用程序可能需要在会话之间保留的临时数据,但不一定需要永久保存。通常使用
Context.getCacheDir()
访问此目录。这将显示为“缓存”在您的应用设置中。
文件
与缓存目录类似,您的应用还具有特定于应用程序的目录以保存文件。在此目录中的文件将存在,直到应用显式删除它们或卸载应用为止。通常使用
Context.getFilesDir()
访问此目录。这可以显示为应用信息屏幕上的各种内容,但在您的截图中,它是“USB存储数据”。
注意:如果要明确放置在外部媒体(通常是SD卡)上,则可以使用
Context.getExternalFilesDir(String type)
。
区别
两个目录都仅适用于您的应用程序(其他应用程序无法访问)。缓存目录和文件目录之间的区别之一是,如果系统存储空间不足,则首先释放资源的地方是您的缓存目录。系统不会清除文件目录中的任何数据。另一个区别是,可以从应用程序信息屏幕手动清除缓存目录。文件目录通常也可以清除,但清除文件目录也将清除缓存目录。
我该使用哪个?
这取决于数据与应用程序生命周期的重要性。如果您只需要在一个会话中使用数据,并且您怀疑永远不需要再次使用该数据,则不要使用任何一个。只需将其保存在内存中,直到不再需要为止。如果您怀疑需要在多个会话之间重复使用数据,但不必保留硬副本,请使用缓存目录。如果您必须无论如何都要使用此数据,或者它是需要持久存储的较大数据,请使用文件目录。以下是我能想到的一些示例:
- 缓存 - 最近打开的电子邮件
- 一旦打开,缓存数据,以便用户再次阅读该电子邮件时,可以立即加载,而无需再次使用网络检索相同的数据。我不需要永久保留这些数据,因为最终用户将完成电子邮件。
- 文件 - 从电子邮件下载的附件
- 这是用户执行的操作,表示“我想保留这些数据,以便在需要时可以重新调用它。”因此,将其放入文件目录中,因为我永远不希望删除此文件,除非用户要求删除。
何时清除缓存目录?
来自Context.getCacheDir()
javadocs:
注意:您不应该依赖系统为您删除这些文件;您应该始终具有合理的最大值,例如1 MB,用于消耗缓存文件的空间,并在超过该空间时修剪这些文件。
它使用了1 MB的示例,但这可能对您的应用程序合理或不合理。无论如何,您都需要设置一个硬性最大值。原因很简单,就是设计一个负责任的应用程序。那么何时检查?我建议每次要将某些内容放入缓存目录时都进行检查。这是一个非常简单的缓存管理器:
public class CacheManager {
private static final long MAX_SIZE = 5242880L;
private CacheManager() {
}
public static void cacheData(Context context, byte[] data, String name) throws IOException {
File cacheDir = context.getCacheDir();
long size = getDirSize(cacheDir);
long newSize = data.length + size;
if (newSize > MAX_SIZE) {
cleanDir(cacheDir, newSize - MAX_SIZE);
}
File file = new File(cacheDir, name);
FileOutputStream os = new FileOutputStream(file);
try {
os.write(data);
}
finally {
os.flush();
os.close();
}
}
public static byte[] retrieveData(Context context, String name) throws IOException {
File cacheDir = context.getCacheDir();
File file = new File(cacheDir, name);
if (!file.exists()) {
return null;
}
byte[] data = new byte[(int) file.length()];
FileInputStream is = new FileInputStream(file);
try {
is.read(data);
}
finally {
is.close();
}
return data;
}
private static void cleanDir(File dir, long bytes) {
long bytesDeleted = 0;
File[] files = dir.listFiles();
for (File file : files) {
bytesDeleted += file.length();
file.delete();
if (bytesDeleted >= bytes) {
break;
}
}
}
private static long getDirSize(File dir) {
long size = 0;
File[] files = dir.listFiles();
for (File file : files) {
if (file.isFile()) {
size += file.length();
}
}
return size;
}
}
当然,这可能是一个昂贵的操作,所以你应该计划在后台线程上缓存。此外,这可以变得像你需要的那样复杂。在我的例子中,我假设所有缓存文件都放置在缓存目录的根目录下,因此我没有检查潜在的子目录。删除文件的例程也可以变得更加复杂,例如按最早访问日期删除文件。决定缓存数据时需要记住的一件事情是,你需要始终为缓存数据不存在的情况做好准备。当你的缓存没有存储数据时,请始终准备一种通过外部手段检索数据的例程。同样,在从外部检索数据之前,请始终检查你的缓存。缓存的目的是减少网络活动、长时间处理并在您的应用程序中提供响应的UI。因此,请负责任地使用它 :)