在回答您的问题之前,这里简要介绍两种存储类型:
快取
这是文件系统上特定于应用程序的目录。该目录的目的是存储临时数据,您的应用程序可能需要在会话之间保留这些临时数据,但对于永久保留这些数据可能并不重要。通常,您可以使用访问此目录Context.getCacheDir()
。这将在您的应用设置上显示为“缓存”。
档案
与缓存目录一样,您的应用程序还具有一个特定于应用程序的目录,用于保存文件。该目录中的文件将一直存在,直到应用程序将其明确删除或将其卸载为止。通常,您可以使用访问此目录Context.getFilesDir()
。这可以显示为应用程序信息屏幕上的各种内容,但是在屏幕快照中,这是“ USB存储数据”。
注意:如果要显式放置在外部介质(通常是SD卡)上,则可以使用Context.getExternalFilesDir(String type)
。
区别
这两个目录仅特定于您的应用程序(其他应用程序无权访问)。缓存和文件目录之间的区别之一是,如果系统存储空间不足,则要释放资源的第一个位置是来自缓存目录。系统不会从文件目录中清除任何数据。另一个区别是,通常可以从应用程序信息屏幕中手动清除缓存目录。文件目录通常也可以,但是清除文件目录也将清除缓存目录。
我要使用哪一个?
这取决于将数据与应用程序生命周期进行比较的重要性。如果您只需要一个会话的数据,并且怀疑是否需要再次使用该数据,则不要使用任何一个。只需将其保留在内存中,直到不需要它为止。如果你怀疑你需要重复使用多个会话之间的数据,但你不具备继续硬拷贝,使用缓存目录。如果无论如何都必须具有此数据,或者需要持久存储的是相当大的数据,请使用files目录。这是我能想到的一些示例:
- 缓存-最近打开的电子邮件
- 打开后,缓存数据,以便当用户想要再次阅读该电子邮件时,它立即加载,而不是再次使用网络来检索相同的数据。我不需要永远保留此地址,因为最终用户将可以使用该电子邮件。
- 文件-从电子邮件下载的附件
- 这是用户说的“我想保留此数据,以便可以在需要时将其备份”的操作。因此,将其放在文件目录中,因为在用户希望删除该文件之前,我永远不想删除该文件。
什么时候应该清除缓存目录?
从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。因此,请负责任地使用它:)