获取应用专属缓存目录

获取Android应用专属缓存存储目录

SD卡缓存目录

当应用需要将图片或者文件缓存到SD卡中时要去申请创建目录,有下面几种途径 
我们可以通过API调用应用专属目录:

// /storage/emulated/0/Android/data/app_package_name/files/Pictures
Content.getExternalFilesDir(Environment.DIRECTORY_PICTURES); 
// /storage/emulated/0/Android/data/app_package_name/cache
Content.getExternalCacheDir(); 
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4

上面两个目录是专属于当前app的,当应用被删除时,上面目录下的文件也会清空

内存缓存目录

相对于应用的专属SD卡缓存有两个内存缓存地址:

Content. getCacheDir(); //  /data/data/app_package_name/cache
Content. getFilesDir(); //  /data/data/app_package_name/files
  • 1
  • 2
  • 1
  • 2

这两个目录中的文件也会随着app的删除而清空

当系统版本大于等于4.4时,对通过上面4个API调用得到的目录进行文件的读写操作不需要申请SD卡的读写权限,所以6.0及以上系统使用时也不需要动态申请读写权限

使用注意事项

  1. 当存储比较大的文件时,如图片等文件存储在SD卡对应的目录下
  2. 应用的内存缓存目录只有应用本身能对其进行读写操作,外部应用不行,如相机应用 (内存目录读写权限:rwxr-x–x,SD卡缓存目录读写权限:rwxrwx—)
  3. 即使是通过自定义路径得到的上述目录,在系统版本大于等于4.4时也不需要申请SD卡读写权限

/**
 * 获取应用专属缓存目录
 * android 4.4及以上系统不需要申请SD卡读写权限
 * 因此也不用考虑6.0系统动态申请SD卡读写权限问题,切随应用被卸载后自动清空 不会污染用户存储空间
 * @param context 上下文
 * @param type 文件夹类型 可以为空,为空则返回API得到的一级目录
 * @return 缓存文件夹 如果没有SD卡或SD卡有问题则返回内存缓存目录,否则优先返回SD卡缓存目录
 */
public static File getCacheDirectory(Context context,String type) {
    File appCacheDir = getExternalCacheDirectory(context,type);
    if (appCacheDir == null){
        appCacheDir = getInternalCacheDirectory(context,type);
    }

    if (appCacheDir == null){
        Log.e("getCacheDirectory","getCacheDirectory fail ,the reason is mobile phone unknown exception !");
    }else {
        if (!appCacheDir.exists()&&!appCacheDir.mkdirs()){
            Log.e("getCacheDirectory","getCacheDirectory fail ,the reason is make directory fail !");
        }
    }
    return appCacheDir;
}

/**
 * 获取SD卡缓存目录
 * @param context 上下文
 * @param type 文件夹类型 如果为空则返回 /storage/emulated/0/Android/data/app_package_name/cache
 *             否则返回对应类型的文件夹如Environment.DIRECTORY_PICTURES 对应的文件夹为 .../data/app_package_name/files/Pictures
 * {@link android.os.Environment#DIRECTORY_MUSIC},
 * {@link android.os.Environment#DIRECTORY_PODCASTS},
 * {@link android.os.Environment#DIRECTORY_RINGTONES},
 * {@link android.os.Environment#DIRECTORY_ALARMS},
 * {@link android.os.Environment#DIRECTORY_NOTIFICATIONS},
 * {@link android.os.Environment#DIRECTORY_PICTURES}, or
 * {@link android.os.Environment#DIRECTORY_MOVIES}.or 自定义文件夹名称
 * @return 缓存目录文件夹 或 null(无SD卡或SD卡挂载失败)
 */
public static File getExternalCacheDirectory(Context context,String type) {
    File appCacheDir = null;
    if( Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())) {
        if (TextUtils.isEmpty(type)){
            appCacheDir = context.getExternalCacheDir();
        }else {
            appCacheDir = context.getExternalFilesDir(type);
        }

        if (appCacheDir == null){// 有些手机需要通过自定义目录
            appCacheDir = new File(Environment.getExternalStorageDirectory(),"Android/data/"+context.getPackageName()+"/cache/"+type);
        }

        if (appCacheDir == null){
            Log.e("getExternalDirectory","getExternalDirectory fail ,the reason is sdCard unknown exception !");
        }else {
            if (!appCacheDir.exists()&&!appCacheDir.mkdirs()){
                Log.e("getExternalDirectory","getExternalDirectory fail ,the reason is make directory fail !");
            }
        }
    }else {
        Log.e("getExternalDirectory","getExternalDirectory fail ,the reason is sdCard nonexistence or sdCard mount fail !");
    }
    return appCacheDir;
}

/**
 * 获取内存缓存目录
 * @param type 子目录,可以为空,为空直接返回一级目录
 * @return 缓存目录文件夹 或 null(创建目录文件失败)
 * 注:该方法获取的目录是能供当前应用自己使用,外部应用没有读写权限,如 系统相机应用
 */
public static File getInternalCacheDirectory(Context context,String type) {
    File appCacheDir = null;
    if (TextUtils.isEmpty(type)){
        appCacheDir = context.getCacheDir();// /data/data/app_package_name/cache
    }else {
        appCacheDir = new File(context.getFilesDir(),type);// /data/data/app_package_name/files/type
    }

    if (!appCacheDir.exists()&&!appCacheDir.mkdirs()){
        Log.e("getInternalDirectory","getInternalDirectory fail ,the reason is make directory fail !");
    }
    return appCacheDir;
}



<think>好的,我现在需要帮助用户解决在安卓开发中获取应用目录路径的问题,并提供示例代码。首先,我要回忆一下之前用户提到的引用内容,虽然那些引用可能不完全相关,但可能有帮助。 用户的问题很明确,他们需要具体的解决方案和代码示例。我记得在Android中,应用目录分为内部存储和外部存储,还有可能涉及缓存目录或者APK路径。首先,内部存储路径通常通过Context的方法获取,比如getFilesDir()和getCacheDir()。外部存储则需要使用Environment.getExternalStorage...相关的方法,但需要注意权限和API版本的变化。 根据引用[1]和[2],用户已经提到过内部存储路径的获取方法,比如context.getFilesDir().getAbsolutePath(),这可能是一个关键点。而引用[3]提到APK路径在/data/app/下,但访问受限,可能需要PackageManager来获取。 接下来,我需要整理常见的几种应用目录路径,并给出对应的代码示例。需要注意不同路径的获取方式以及所需的权限。比如,内部存储不需要权限,而外部存储可能需要READ_EXTERNAL_STORAGE权限,特别是在较新的Android版本中。 另外,用户可能需要了解这些路径的区别和使用场景。例如,内部存储是应用私有的,外部存储可能分为私有目录和公共目录,公共目录需要权限,而私有目录在Android 4.4及以上不需要权限。 还需要考虑代码的兼容性,比如使用ContextCompat.getExternalFilesDirs()来获取外部存储的路径,包括可能存在的SD卡路径。同时,对于缓存目录,getCacheDir()和getExternalCacheDir()的区别也需要说明。 最后,关于APK路径,虽然用户没有明确提到,但引用[3]提到使用PackageManager,所以可能需要包括如何通过PackageManager获取应用信息,进而得到APK路径,但需要注意权限问题,比如需要REQUEST_INSTALL_PACKAGES权限。 现在需要把这些信息结构化,分点说明,并给出对应的代码示例。确保代码正确,使用中文注释,并提醒用户注意权限和API版本差异。可能还需要在注意事项中提到存储访问框架(SAF)或者Scoped Storage的影响,特别是在Android 10及以上版本。 总结步骤: 1. 内部存储目录:getFilesDir()和getCacheDir() 2. 外部私有目录:getExternalFilesDir()和getExternalCacheDir() 3. 公共存储目录:Environment.getExternalStoragePublicDirectory()(已过时,需注意) 4. APK安装路径:通过PackageManager获取ApplicationInfo的sourceDir 5. 提醒权限和版本差异 6. 提供完整示例代码,并解释各部分的作用 需要检查是否有遗漏的路径,比如应用的数据库路径或者其他特殊目录。同时,确保代码示例中的方法都是当前推荐使用的,避免使用已弃用的API。 可能还需要提到Context的方法,如getDataDir()获取应用的数据目录,这在API 24+可用。不过对于兼容性,需要判断版本或使用其他方法。 现在将这些内容整理成清晰的部分,每部分说明用途、代码示例和注意事项,最后汇总一个完整的示例,并附上相关问题。</think>### 安卓应用目录路径获取方法及代码示例 #### 一、内部存储路径 1. **私有文件目录** - **用途**:存储应用私有文件,其他应用无法直接访问 - **代码**: ```java String filesPath = getApplicationContext().getFilesDir().getAbsolutePath(); // 示例输出:/data/user/0/包名/files ``` 2. **缓存目录** - **用途**:存储临时缓存文件,系统可能自动清理 - **代码**: ```java String cachePath = getApplicationContext().getCacheDir().getAbsolutePath(); // 示例输出:/data/user/0/包名/cache ``` --- #### 二、外部存储路径 1. **外部私有目录** - **用途**:应用专属的外部存储目录,Android 4.4+ 无需权限 - **代码**: ```java // 获取主外部存储路径 File externalFilesDir = getApplicationContext().getExternalFilesDir(null); String externalPath = externalFilesDir.getAbsolutePath(); // 示例输出:/storage/emulated/0/Android/data/包名/files // 获取外部缓存目录 File externalCacheDir = getApplicationContext().getExternalCacheDir(); String externalCachePath = externalCacheDir.getAbsolutePath(); ``` 2. **公共目录**(需权限) - **用途**:访问系统公共目录(如Downloads、DCIM) - **代码**(API ≤ 28): ```java File publicDownloads = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS); // 需添加权限:<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> ``` --- #### 三、APK安装路径 - **用途**:获取当前应用的APK文件路径 - **代码**: ```java try { String apkPath = getPackageManager().getApplicationInfo(getPackageName(), 0).sourceDir; // 示例输出:/data/app/包名/base.apk } catch (PackageManager.NameNotFoundException e) { e.printStackTrace(); } ``` --- #### 四、完整示例代码 ```java // 内部存储路径 String internalFilesPath = getFilesDir().getAbsolutePath(); String internalCachePath = getCacheDir().getAbsolutePath(); // 外部存储路径 File externalFilesDir = getExternalFilesDir(Environment.DIRECTORY_DOCUMENTS); String externalFilesPath = externalFilesDir != null ? externalFilesDir.getPath() : "不可用"; // 获取APK路径 String apkPath; try { apkPath = getPackageManager().getApplicationInfo(getPackageName(), 0).sourceDir; } catch (Exception e) { apkPath = "获取失败"; } // 打印结果 Log.d("PathInfo", "内部文件路径:" + internalFilesPath + "\n" + "内部缓存路径:" + internalCachePath + "\n" + "外部文档路径:" + externalFilesPath + "\n" + "APK路径:" + apkPath); ``` --- #### 注意事项 1. **权限声明**: ```xml <!-- 访问外部公共目录需要声明权限 --> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> ``` 2. **API兼容性**: - `Environment.getExternalStoragePublicDirectory()` 在Android 10(API 29)后已废弃,建议改用[MediaStore API](https://developer.android.com/training/data-storage/shared/media)或[存储访问框架](https://developer.android.com/guide/topics/providers/document-provider) 3. **多用户环境**: 使用`ContextCompat.getExternalFilesDirs()`可获取所有外部存储设备路径(包括SD卡) ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值