Android получает свободный размер внутренней / внешней памяти

98

Я хочу программно получить размер свободной памяти на внутреннем / внешнем накопителе моего устройства. Я использую этот фрагмент кода:

StatFs stat = new StatFs(Environment.getExternalStorageDirectory().getPath());
long bytesAvailable = (long)stat.getBlockSize() *(long)stat.getBlockCount();
long megAvailable = bytesAvailable / 1048576;
Log.e("","Available MB : "+megAvailable);

File path = Environment.getDataDirectory();
StatFs stat2 = new StatFs(path.getPath());
long blockSize = stat2.getBlockSize();
long availableBlocks = stat2.getAvailableBlocks();
String format =  Formatter.formatFileSize(this, availableBlocks * blockSize);
Log.e("","Format : "+format);

и результат, который я получаю:

11-15 10:27:18.844: E/(25822): Available MB : 7572
11-15 10:27:18.844: E/(25822): Format : 869MB

Проблема в том, что я хочу получить свободную память SdCard, которая есть 1,96GBпрямо сейчас. Как я могу исправить этот код, чтобы получить бесплатный размер?

Android-дроид
источник
Начиная с уровня API 18, они переименовали метод в конец на Long. Вероятно, вам нужно будет добавить проверку уровня API перед этим
Джейшил Дэйв
Все решения, которые я пробовал, ни с кем не работал, когда я выполняю форматирование как внутреннее хранилище ... вы можете меня порадовать, как этого добиться?
Йогеш Рати

Ответы:

182

Ниже приведен код для вашей цели:

public static boolean externalMemoryAvailable() {
        return android.os.Environment.getExternalStorageState().equals(
                android.os.Environment.MEDIA_MOUNTED);
    }

    public static String getAvailableInternalMemorySize() {
        File path = Environment.getDataDirectory();
        StatFs stat = new StatFs(path.getPath());
        long blockSize = stat.getBlockSizeLong();
        long availableBlocks = stat.getAvailableBlocksLong();
        return formatSize(availableBlocks * blockSize);
    }

    public static String getTotalInternalMemorySize() {
        File path = Environment.getDataDirectory();
        StatFs stat = new StatFs(path.getPath());
        long blockSize = stat.getBlockSizeLong();
        long totalBlocks = stat.getBlockCountLong();
        return formatSize(totalBlocks * blockSize);
    }

    public static String getAvailableExternalMemorySize() {
        if (externalMemoryAvailable()) {
            File path = Environment.getExternalStorageDirectory();
            StatFs stat = new StatFs(path.getPath());
            long blockSize = stat.getBlockSizeLong();
            long availableBlocks = stat.getAvailableBlocksLong();
            return formatSize(availableBlocks * blockSize);
        } else {
            return ERROR;
        }
    }

    public static String getTotalExternalMemorySize() {
        if (externalMemoryAvailable()) {
            File path = Environment.getExternalStorageDirectory();
            StatFs stat = new StatFs(path.getPath());
            long blockSize = stat.getBlockSizeLong();
            long totalBlocks = stat.getBlockCountLong();
            return formatSize(totalBlocks * blockSize);
        } else {
            return ERROR;
        }
    }

    public static String formatSize(long size) {
        String suffix = null;

        if (size >= 1024) {
            suffix = "KB";
            size /= 1024;
            if (size >= 1024) {
                suffix = "MB";
                size /= 1024;
            }
        }

        StringBuilder resultBuffer = new StringBuilder(Long.toString(size));

        int commaOffset = resultBuffer.length() - 3;
        while (commaOffset > 0) {
            resultBuffer.insert(commaOffset, ',');
            commaOffset -= 3;
        }

        if (suffix != null) resultBuffer.append(suffix);
        return resultBuffer.toString();
    }

Получить размер RAM

ActivityManager actManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
MemoryInfo memInfo = new ActivityManager.MemoryInfo();
actManager.getMemoryInfo(memInfo);
long totalMemory = memInfo.totalMem;
Динеш Праджапати
источник
2
getBlockSize()и getBlockCountустарели.
Nima G
2
@DineshPrajapati Спасибо за ответ, у меня есть запрос. Если я использую Environment.getRootDirectory () вместо Environment.getDataDirectory для вычисления внутреннего хранилища, я получаю некоторый результат .. это относится к внутренней памяти, другой памяти ..
AK Joshi
3
@DineshPrajapati .. Проверено на MOTO G2. Неправильные данные для внешнего хранилища
AK Joshi
1
Может случиться так, что пара API устареет, проверьте это на developers.android.com
Динеш Праджапати,
1
Используйте Long в конце для новых уровней API (> 18)
Gun2sh
40

Вот как я это сделал:

StatFs stat = new StatFs(Environment.getExternalStorageDirectory().getPath());
long bytesAvailable;
if (android.os.Build.VERSION.SDK_INT >= 
    android.os.Build.VERSION_CODES.JELLY_BEAN_MR2) {
    bytesAvailable = stat.getBlockSizeLong() * stat.getAvailableBlocksLong();
}
else {
    bytesAvailable = (long)stat.getBlockSize() * (long)stat.getAvailableBlocks();
}
long megAvailable = bytesAvailable / (1024 * 1024);
Log.e("","Available MB : "+megAvailable);
Android-дроид
источник
3
но это испорчено :(
abbasalim
@ ArMo372, вы ребята узнали код замены для этого?
SimpleCoder
3
Просто замените getBlockSizeи getAvailableBlocksна getBlockSizeLongи getAvailableBlocksLong.
smg
1
это не получает правильного доступного места. Это 1141 вместо 1678 @smg
1
Решение не работает, когда я выполняю форматирование как внутреннее хранилище ... Не могли бы вы доставить мне удовольствие, как это сделать?
Йогеш Рати,
27

Начиная с API 9 вы можете:

long freeBytesInternal = new File(ctx.getFilesDir().getAbsoluteFile().toString()).getFreeSpace();
long freeBytesExternal = new File(getExternalFilesDir(null).toString()).getFreeSpace();
Цойкер
источник
2
File.getUsableSpace (), вероятно, лучше, потому что вы, вероятно, не работаете с правами root.
Марк
В File.getUsableSpace()выглядит как простой метод использовать , а не использовать StatFs. Зачем мне использовать StatFs@MarkCarter?
StuStirling
1
@ DiscoS2 Вы бы использовали StatFs, если ваш minSdkVersion меньше 9.
Марк,
1
Как бы вы также отслеживали изменения хранилища?
разработчик Android
24

Чтобы получить все доступные папки хранилища (включая SD-карты), вы сначала получите файлы хранилища:

File internalStorageFile=getFilesDir();
File[] externalStorageFiles=ContextCompat.getExternalFilesDirs(this,null);

Затем вы можете получить доступный размер каждого из них.

Сделать это можно тремя способами:

API 8 и ниже:

StatFs stat=new StatFs(file.getPath());
long availableSizeInBytes=stat.getBlockSize()*stat.getAvailableBlocks();

API 9 и выше:

long availableSizeInBytes=file.getFreeSpace();

API 18 и выше (не требуется, если подходит предыдущий):

long availableSizeInBytes=new StatFs(file.getPath()).getAvailableBytes(); 

Чтобы получить красивую отформатированную строку того, что у вас есть сейчас, вы можете использовать:

String formattedResult=android.text.format.Formatter.formatShortFileSize(this,availableSizeInBytes);

или вы можете использовать это, если хотите увидеть точное количество байтов, но красиво:

NumberFormat.getInstance().format(availableSizeInBytes);

Обратите внимание, что я думаю, что внутреннее хранилище может быть таким же, как первое внешнее хранилище, поскольку первое является эмулированным.


РЕДАКТИРОВАТЬ: Используя StorageVolume на Android Q и выше, я думаю, что можно получить свободное место для каждого, используя что-то вроде:

    val storageManager = getSystemService(Context.STORAGE_SERVICE) as StorageManager
    val storageVolumes = storageManager.storageVolumes
    AsyncTask.execute {
        for (storageVolume in storageVolumes) {
            val uuid: UUID = storageVolume.uuid?.let { UUID.fromString(it) } ?: StorageManager.UUID_DEFAULT
            val allocatableBytes = storageManager.getAllocatableBytes(uuid)
            Log.d("AppLog", "allocatableBytes:${android.text.format.Formatter.formatShortFileSize(this,allocatableBytes)}")
        }
    }

Я не уверен, что это правильно, и я не могу найти способ получить общий размер каждого, поэтому я написал об этом здесь и спросил об этом здесь .

разработчик Android
источник
1
Как получить свободное место на съемной SD-карте (или флешке USB OTG) на устройствах с API 23? new StatFs (file.getPath ()). getAvailableBytes () или file.getUsableSpace () дает 972546048 байт независимо от реального размера хранилища на Nexus 5 (Marshmallow 6.0.1).
04
@isabsent Nexus 5 не имеет слота для SD-карты. Как вы это проверяли?
разработчик Android
Проверил с флешкой USB OTG.
05
@isabsent Я никогда этим не пользовался. Сожалею. Хорошо ли работает на API 22 и ниже?
разработчик Android
1
@Smeet Можно ли попробовать на Android 6 или выше? Если да, то, возможно, проблема в следующем: code.google.com/p/android/issues/detail?id=200326
разработчик Android,
9

@ Android-Droid - вы неправильно Environment.getExternalStorageDirectory()указываете на внешнее хранилище, которое не обязательно должно быть SD-картой, оно также может быть монтировкой внутренней памяти. Видеть:

Найдите расположение внешней SD-карты

Sharp80
источник
7

Попробуйте этот простой фрагмент

    public static String readableFileSize() {
    long availableSpace = -1L;
    StatFs stat = new StatFs(Environment.getExternalStorageDirectory().getPath());
    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2)
        availableSpace = (long) stat.getBlockSizeLong() * (long) stat.getAvailableBlocksLong();
    else
        availableSpace = (long) stat.getAvailableBlocks() * (long) stat.getBlockSize();

    if(availableSpace <= 0) return "0";
    final String[] units = new String[] { "B", "kB", "MB", "GB", "TB" };
    int digitGroups = (int) (Math.log10(availableSpace)/Math.log10(1024));
    return new DecimalFormat("#,##0.#").format(availableSpace/Math.pow(1024, digitGroups)) + " " + units[digitGroups];
}
Несс Тяги
источник
Спасибо, но у меня java.lang.ArrayIndexOutOfBoundsException: length=5; index=-2147483648ошибка, кажется, digitGroupsрезультат -2147483648.
Акуна
Решение не работает, когда я выполняю форматирование как внутреннее хранилище ... Не могли бы вы меня порадовать, как этого добиться?
Йогеш Рати,
6

Очень легко узнать доступное хранилище, если у вас есть внутренний и внешний путь хранения. Кроме того, внешний путь к памяти телефона действительно очень легко узнать, используя

Environment.getExternalStorageDirectory (). GetPath ();

Поэтому я просто концентрируюсь на том, как узнать пути внешнего съемного хранилища, такого как съемная SD-карта, USB OTG (не тестировался USB OTG, так как у меня нет USB OTG).

Ниже приведен список всех возможных путей внешнего съемного хранилища.

 /**
     * This method returns the list of removable storage and sdcard paths.
     * I have no USB OTG so can not test it. Is anybody can test it, please let me know
     * if working or not. Assume 0th index will be removable sdcard path if size is
     * greater than 0.
     * @return the list of removable storage paths.
     */
    public static HashSet<String> getExternalPaths()
    {
    final HashSet<String> out = new HashSet<String>();
    String reg = "(?i).*vold.*(vfat|ntfs|exfat|fat32|ext3|ext4).*rw.*";
    String s = "";
    try
    {
        final Process process = new ProcessBuilder().command("mount").redirectErrorStream(true).start();
        process.waitFor();
        final InputStream is = process.getInputStream();
        final byte[] buffer = new byte[1024];
        while (is.read(buffer) != -1)
        {
            s = s + new String(buffer);
        }
        is.close();
    }
    catch (final Exception e)
    {
        e.printStackTrace();
    }

    // parse output
    final String[] lines = s.split("\n");
    for (String line : lines)
    {
        if (!line.toLowerCase(Locale.US).contains("asec"))
        {
            if (line.matches(reg))
            {
                String[] parts = line.split(" ");
                for (String part : parts)
                {
                    if (part.startsWith("/"))
                    {
                        if (!part.toLowerCase(Locale.US).contains("vold"))
                        {
                            out.add(part.replace("/media_rw","").replace("mnt", "storage"));
                        }
                    }
                }
            }
        }
    }
    //Phone's external storage path (Not removal SDCard path)
    String phoneExternalPath = Environment.getExternalStorageDirectory().getPath();

    //Remove it if already exist to filter all the paths of external removable storage devices
    //like removable sdcard, USB OTG etc..
    //When I tested it in ICE Tab(4.4.2), Swipe Tab(4.0.1) with removable sdcard, this method includes
    //phone's external storage path, but when i test it in Moto X Play (6.0) with removable sdcard,
    //this method does not include phone's external storage path. So I am going to remvoe the phone's
    //external storage path to make behavior consistent in all the phone. Ans we already know and it easy
    // to find out the phone's external storage path.
    out.remove(phoneExternalPath);

    return out;
}
Смит
источник
Насколько я помню, использование постоянных имен для обработки путей может не работать на некоторых устройствах, поскольку у некоторых могут быть свои собственные пути. Я надеюсь, что это не так. +1 за усилия.
разработчик Android,
1
@androiddeveloper Спасибо за голосование. Мне нужна поддержка всех, чтобы протестировать этот код на вашем устройстве, потому что у меня есть не все устройства, но они протестированы на 4 разных устройствах и работают нормально. Прокомментируйте, пожалуйста, здесь не работает ни в одном мобиле тела.
Smeet
Решение не работает, когда я выполняю форматирование как внутреннее хранилище ... Не могли бы вы доставить мне удовольствие, как это сделать?
Йогеш Рати,
4

Быстрое добавление в тему Внешняя память

Пусть вас не смущает название метода externalMemoryAvailable()в ответе Динеш Праджапати.

Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())дает вам текущее состояние памяти, если носитель присутствует и смонтирован в своей точке монтирования с доступом для чтения / записи. Получится trueдаже на устройствах без SD-карт, вроде Nexus 5. Но все же это метод «must-have» перед любыми операциями с хранилищем.

Чтобы проверить, есть ли на вашем устройстве SD-карта, вы можете использовать метод ContextCompat.getExternalFilesDirs()

Он не показывает переходные устройства, такие как USB-накопители.

Также имейте в виду, что ContextCompat.getExternalFilesDirs()на Android 4.3 и ниже всегда будет возвращаться только 1 запись (SD-карта, если она доступна, в противном случае внутренняя). Вы можете прочитать об этом здесь .

  public static boolean isSdCardOnDevice(Context context) {
    File[] storages = ContextCompat.getExternalFilesDirs(context, null);
    if (storages.length > 1 && storages[0] != null && storages[1] != null)
        return true;
    else
        return false;
}

в моем случае этого было достаточно, но не забывайте, что на некоторых устройствах Android может быть 2 SD-карты, поэтому, если вам нужны все они - скорректируйте код выше.

Кирилл Кармазин
источник
2
@RequiresApi(api = Build.VERSION_CODES.O)
private void showStorageVolumes() {
    StorageStatsManager storageStatsManager = (StorageStatsManager) getSystemService(Context.STORAGE_STATS_SERVICE);
    StorageManager storageManager = (StorageManager) getSystemService(Context.STORAGE_SERVICE);
    if (storageManager == null || storageStatsManager == null) {
        return;
    }
    List<StorageVolume> storageVolumes = storageManager.getStorageVolumes();
    for (StorageVolume storageVolume : storageVolumes) {
        final String uuidStr = storageVolume.getUuid();
        final UUID uuid = uuidStr == null ? StorageManager.UUID_DEFAULT : UUID.fromString(uuidStr);
        try {
            Log.d("AppLog", "storage:" + uuid + " : " + storageVolume.getDescription(this) + " : " + storageVolume.getState());
            Log.d("AppLog", "getFreeBytes:" + Formatter.formatShortFileSize(this, storageStatsManager.getFreeBytes(uuid)));
            Log.d("AppLog", "getTotalBytes:" + Formatter.formatShortFileSize(this, storageStatsManager.getTotalBytes(uuid)));
        } catch (Exception e) {
            // IGNORED
        }
    }
}

Класс StorageStatsManager представил Android O и выше, который может предоставить вам свободный и общий байт во внешнем / внутреннем хранилище. Подробнее с исходным кодом вы можете прочитать в моей следующей статье. вы можете использовать отражение для ниже, чем Android O

https://medium.com/cashify-engineering/how-to-get-storage-stats-in-android-o-api-26-4b92eca6805b

Бриеш Гупта
источник
2

Так я и сделал ..

внутренняя Общая память

double totalSize = new File(getApplicationContext().getFilesDir().getAbsoluteFile().toString()).getTotalSpace();
double totMb = totalSize / (1024 * 1024);

Внутренний свободный размер

 double availableSize = new File(getApplicationContext().getFilesDir().getAbsoluteFile().toString()).getFreeSpace();
    double freeMb = availableSize/ (1024 * 1024);

Внешняя свободная и общая память

 long freeBytesExternal =  new File(getExternalFilesDir(null).toString()).getFreeSpace();
       int free = (int) (freeBytesExternal/ (1024 * 1024));
        long totalSize =  new File(getExternalFilesDir(null).toString()).getTotalSpace();
        int total= (int) (totalSize/ (1024 * 1024));
       String availableMb = free+"Mb out of "+total+"MB";
маквине
источник
0

По поводу внешней менории есть еще один способ:
File external = Environment.getExternalStorageDirectory(); free:external.getFreeSpace(); total:external.getTotalSpace();

Эдвард Андерсон
источник
0

После проверки другого решения напишите код сам, это полный код для поиска

  • Общая внешняя память
  • Свободная внешняя память
  • Используемая внешняя память
  • Общая внутренняя память
  • Используемая внутренняя память
  • Свободная внутренняя память

'' ''

object DeviceMemoryUtil {
private const val error: String = "Something went wrog"
private const val noExternalMemoryDetected = "No external Storage detected"
private var totalExternalMemory: Long = 0
private var freeExternalMemory: Long = 0
private var totalInternalStorage: Long = 0
private var freeInternalStorage: Long = 0

/**
 * Checks weather external memory is available or not
 */
private fun externalMemoryAvailable(): Boolean {
    return Environment.getExternalStorageState() ==
            Environment.MEDIA_MOUNTED
}

/**
 *Gives total external memory
 * @return String Size of external memory
 * @return Boolean True if memory size is returned
 */
fun getTotalExternalMemorySize(): Pair<String?, Boolean> {
    val dirs: Array<File> = ContextCompat.getExternalFilesDirs(CanonApplication.getCanonAppInstance(), null)
    return if (externalMemoryAvailable()) {
        if (dirs.size > 1) {
            val stat = StatFs(dirs[1].path)
            val blockSize = stat.blockSizeLong
            val totalBlocks = stat.blockCountLong
            var totalExternalSize = totalBlocks * blockSize
            totalExternalMemory = totalExternalSize
            Pair(formatSize(totalExternalSize), true)
        } else {
            Pair(error, false)
        }
    } else {
        Pair(noExternalMemoryDetected, false)
    }
}

/**
 * Gives free external memory size
 * @return String Size of free external memory
 * @return Boolean True if memory size is returned
 */
fun getAvailableExternalMemorySize(): Pair<String?, Boolean> {
    val dirs: Array<File> = ContextCompat.getExternalFilesDirs(CanonApplication.getCanonAppInstance(), null)
    if (externalMemoryAvailable()) {
        return if (dirs.size > 1) {
            val stat = StatFs(dirs[1].path)
            val blockSize = stat.blockSizeLong
            val availableBlocks = stat.availableBlocksLong
            var freeExternalSize = blockSize * availableBlocks
            freeExternalMemory = freeExternalSize
            Pair(formatSize(freeExternalSize), true)
        } else {
            Pair(error, false)
        }
    } else {
        return Pair(noExternalMemoryDetected, false)
    }
}

/**
 * Gives used external memory size
 *  @return String Size of used external memory
 * @return Boolean True if memory size is returned
 */
fun getUsedExternalMemorySize(): Pair<String?, Boolean> {
    return if (externalMemoryAvailable()) {
        val totalExternalSize = getTotalExternalMemorySize()
        val freeExternalSize = getAvailableExternalMemorySize()
        if (totalExternalSize.second && freeExternalSize.second) {
            var usedExternalVolume = totalExternalMemory - freeExternalMemory
            Pair(formatSize(usedExternalVolume), true)
        } else {
            Pair(error, false)
        }
    } else {
        Pair(noExternalMemoryDetected, false)
    }
}

/**
 *Formats the long to size of memory in gb,mb etc.
 * @param size Size of memory
 */
fun formatSize(size: Long): String? {
    return android.text.format.Formatter.formatFileSize(CanonApplication.getCanonAppInstance(), size)
}

/**
 * Gives total internal memory size
 *  @return String Size of total internal memory
 * @return Boolean True if memory size is returned
 */
fun getTotalInternalStorage(): Pair<String?, Boolean> {
    if (showStorageVolumes()) {
        return Pair(formatSize(totalInternalStorage), true)
    } else {
        return Pair(error, false)
    }

}

/**
 * Gives free or available internal memory size
 *  @return String Size of free internal memory
 * @return Boolean True if memory size is returned
 */
fun getFreeInternalStorageVolume(): Pair<String?, Boolean> {
    return if (showStorageVolumes()) {
        Pair(formatSize(freeInternalStorage), true)
    } else {
        Pair(error, false)
    }
}

/**
 *For calculation of internal storage
 */
private fun showStorageVolumes(): Boolean {
    val storageManager = CanonApplication.canonApplicationInstance.applicationContext.getSystemService(Context.STORAGE_SERVICE) as StorageManager
    val storageStatsManager = CanonApplication.canonApplicationInstance.applicationContext.getSystemService(Context.STORAGE_STATS_SERVICE) as StorageStatsManager
    if (storageManager == null || storageStatsManager == null) {
        return false
    }
    val storageVolumes: List<StorageVolume> = storageManager.storageVolumes
    for (storageVolume in storageVolumes) {
        var uuidStr: String? = null
        storageVolume.uuid?.let {
            uuidStr = it
        }
        val uuid: UUID = if (uuidStr == null) StorageManager.UUID_DEFAULT else UUID.fromString(uuidStr)
        return try {
            freeInternalStorage = storageStatsManager.getFreeBytes(uuid)
            totalInternalStorage = storageStatsManager.getTotalBytes(uuid)
            true
        } catch (e: Exception) {
            // IGNORED
            false
        }
    }
    return false
}

fun getTotalInternalExternalMemory(): Pair<Long?, Boolean> {
    if (externalMemoryAvailable()) {
        if (getTotalExternalMemorySize().second) {
            if (getTotalInternalStorage().second) {
                return Pair(totalExternalMemory + totalInternalStorage, true)
            } else {
                return Pair(0, false)
            }
        }
        return Pair(0, false)
    } else {
        if (getTotalInternalStorage().second) {
            return Pair(totalInternalStorage, true)
        } else {
            return Pair(0, false)
        }
    }

}

fun getTotalFreeStorage(): Pair<Long,Boolean> {
    if (externalMemoryAvailable()){
        if(getFreeInternalStorageVolume().second){
            getFreeInternalStorageVolume()
            getAvailableExternalMemorySize()
                return Pair(freeExternalMemory + freeInternalStorage,true)
        }
        else{
            return Pair(0,false)
        }
    }
    else {
        if (getFreeInternalStorageVolume().second){
            getFreeInternalStorageVolume()
            return Pair(freeInternalStorage,true)
        }
      else{
            return Pair(0,false)
        }
    }

}}
Сахил Бансал
источник