【问题标题】:When to clear the cache dir in Android?何时清除 Android 中的缓存目录?
【发布时间】:2012-04-14 02:44:24
【问题描述】:

我有一个显示来自互联网的图片的应用程序(设计师作品展示)。我开始在内部缓存目录中缓存我的内容,但应用程序内容可能需要大约 150 MB 的缓存大小。以及 android 文档所说的:

您应该始终自己维护缓存文件并保持在一个 合理的空间消耗限制,例如 1MB。当用户 卸载您的应用程序,这些文件将被删除。

所以我查看了 Currents 应用程序 (Galaxy Nexus),该应用程序的缓存大小为 110 MB。但奇怪的是,像 Google Currents 和 Google Maps 这样的应用程序会将内容缓存在一个名为(USB 存储数据)的东西中:

那么,之前的应用程序使用的这个“USB 存储数据”是什么?如果你在你的应用程序中实现缓存,每次你需要插入一些东西然后比较和清除它时,你是否遍历所有your application files in cache 来获取大小?还是你会一直缓存内容,直到 Android 决定清理某些应用程序缓存目录的时间?

我真的很想知道在 Android 中管理缓存的流程是什么,或者至少其他应用程序如何处理要缓存的大内容。

【问题讨论】:

  • 我认为USB存储数据是保存在SD中的缓存。在我看来,缓存的大小应该由最终用户在偏好活动中选择,删除最后访问的文件以保持限制。
  • 计算机科学中只有两个难点:缓存失效和命名。 -- Phil Karlton
  • @msanford 你知道什么更难吗? 正在关闭 vim...

标签: android caching


【解决方案1】:

在我回答你的问题之前,先简单解释一下这两种存储类型:

缓存

这是文件系统上特定于应用程序的目录。此目录的目的是存储您的应用程序可能需要在会话之间保留的临时数据,但对于永久保留它们可能并不重要。您通常使用Context.getCacheDir() 访问此目录。这将在您的应用设置中显示为“缓存”。

文件

与缓存目录一样,您的应用也有一个特定于应用的目录来保存文件。此目录中的文件将一直存在,直到应用程序明确删除它们或应用程序被卸载。您通常使用Context.getFilesDir() 访问此目录。这可以在应用程序信息屏幕上显示为各种内容,但在您的屏幕截图中,这是“USB 存储数据”。

注意:如果您想明确放置在外部媒体(通常是 SD 卡)上,您可以使用 Context.getExternalFilesDir(String type)

区别

这两个目录仅特定于您的应用程序(其他应用程序无权访问)。缓存目录和文件目录之间的区别之一是,如果系统存储空间不足,它首先要释放资源的地方是缓存目录。系统不会清除文件目录中的任何数据。另一个区别是缓存目录通常可以从应用程序信息屏幕手动清除。 files 目录通常也可以,但是清除 files 目录也会清除缓存目录。

我用哪一个?

这取决于数据与应用生命周期相比的重要性。如果您只需要一个会话的数据并且您怀疑您是否需要再次使用该数据,那么不要使用任何一个。只需将其保存在内存中,直到您不需要它为止。如果您怀疑需要在多个会话之间重用数据,但您没有 保留硬拷贝,请使用缓存目录。如果无论如何您都必须拥有这些数据,或者如果它是需要持久存储的相当大的数据,请使用 files 目录。以下是我能想到的一些例子:

  • 缓存 - 最近打开的电子邮件
    • 一旦打开,缓存数据,这样当用户想要再次阅读该电子邮件时,它会立即加载,而不是再次使用网络来检索相同的数据。我不需要永远保留它,因为最终用户将完成电子邮件。
  • 文件 - 从电子邮件下载的附件
    • 这是用户执行的操作,他说“我想保留此数据,以便在需要时将其拉回”。因此,将其放在文件目录中,因为我不想删除此文件,直到用户希望将其删除。

什么时候应该清除缓存目录?

来自Context.getCacheDir()javadocs:

注意:您不应依赖系统为您删除这些文件; 您应该始终有一个合理的最大值,例如 1 MB, 缓存文件占用的空间量,并修剪这些文件 当超过那个空间时。

它使用 1 MB 的示例,但这对于您的应用来说可能合理,也可能不合理。无论如何,您需要设置一个硬最大值。其原因归结为设计一个负责任的应用程序。那么什么时候应该检查呢?我建议每次你想在缓存目录中放一些东西时检查一下。这是一个非常简单的缓存管理器:

public class CacheManager {

    private static final long MAX_SIZE = 5242880L; // 5MB

    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()) {
            // Data doesn't exist
            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。所以负责任地使用它:)

【讨论】:

  • 如果我有一个需要软存储大文件的应用程序怎么办?我是否应该将缓存设置得非常高并且不因规范或存储在 filesDir 中而负责并找到从那里删除文件的方法?
  • @hipkiss 缓存的大小会因应用程序的要求而异。将最大值设置为您认为对您的应用来说合理的值。
  • @JasonRobinson 是否有缓存目录的最大大小限制,或者应用程序可以在那里保存尽可能多的数据(比如 500 MB)?
  • @Diffy 据我所知没有,但我不确定。
  • @JasonRobinson 我有一个问题要问。我正在构建一个需要从网络服务器访问大型 pdf 或 epub 文件的电子图书馆应用程序。我应该使用什么缓存目录或文件?谢谢。
【解决方案2】:

我是在活动完成时清除应用程序缓存的最佳方法,以便每次调用新活动时都清除缓存。

将此代码放入 onDestroy() 以清除应用缓存

@Override
protected void onDestroy() {

    super.onDestroy();
    try {
        trimCache(this);
       // Toast.makeText(this,"onDestroy " ,Toast.LENGTH_LONG).show();
    } catch (Exception e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

public static void trimCache(Context context) {
    try {
       File dir = context.getCacheDir();
       if (dir != null && dir.isDirectory()) {
          deleteDir(dir);
       }
    } catch (Exception e) {
       // TODO: handle exception
    }
 }

public static boolean deleteDir(File dir) {
    if (dir != null && dir.isDirectory()) {
       String[] children = dir.list();
       for (int i = 0; i < children.length; i++) {
          boolean success = deleteDir(new File(dir, children[i]));
          if (!success) {
             return false;
          }
       }
    }
    // The directory is now empty so delete it
    return dir.delete();
}

【讨论】:

  • onDestroy() supercall 之后?你是srs吗?
【解决方案3】:

我认为缓存背后的想法是在上面写任何你想要的东西,如果它变得太高,Android 会管理它的大小。

您应该记住,您可以将文件写入缓存,但在尝试访问文件时始终检查文件是否仍然保存。并让android管理缓存。

【讨论】:

    【解决方案4】:

    取决于应用程序的类型:

    • 有些应用程序只使用单个会话,不需要记住任何数据,因此您可以随时清除缓存(有些应用程序甚至在其 onStop 活动中自动执行此操作)
    • 大多数应用程序会保留您的数据,因为它们会记住您的设置、您用于登录的帐户……在这种情况下,最好仅在您不经常使用该应用程序时清除缓存。李>

    还有:

    So i took a look at Chrome app (Galaxy Nexus) and the cache size for the application is 110 MB. But what wired is that applications like Google current & Google maps cache the content in something called (USB Storage Data) :
    

    AFAIK,Usb 存储数据与缓存的用途不同:存储用于存储程序特定信息(如 GPS 应用程序的地图),缓存用于存储用户特定信息(如登录)

    如果是谷歌地图:我假设他们将地图数据存储在 USB 存储中,并将您的设置和搜索历史记录保存在缓存中 ==> 地图数据是应用程序特定的,设置和搜索历史是用户特定的

    【讨论】:

      【解决方案5】:

      根据文档,the device is low on internal storage 时系统将清除缓存。自 API8 以来,您拥有 getExternalCacheDir() 方法,我认为该方法很有用,因为我读过您可以拥有大约 150MB 的数据,但外部缓存的缺点是,如果缓存目录太大,您将不得不自己清理缓存目录。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2017-07-29
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2011-10-17
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多