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

問題は、1,96GB現在のSdCardの空きメモリを取得したいことです。このコードを修正して空きサイズを取得するにはどうすればよいですか?


APIレベル18以降、メソッドの名前をLongで終わるように変更しました。おそらく、その前にAPIレベルのチェックを追加する必要があります
Jayshil Dave

すべての解決策内部ストレージとしてフォーマットするとき、誰も動かさないようにしました...どうすればこれを達成できますか?
Yogesh Rathi

回答:


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
内部ストレージを計算するためのEnvironment.getDataDirectory代わり-の@DineshPrajapati私はEnvironment.getRootDirectoryを(使用する場合の答えのおかげで、私は、クエリを持っている)、私はいくつかの出力が..this取得していますと、内部メモリ、他のメモリを指し...
AKジョシ

3
@DineshPrajapati .. MOTO G2でテスト済み外部ストレージの誤ったデータを取得
AK Joshi

1
新しいAPIレベル(> 18)には、最後にLongを使用します
Gun2sh

1
知識を共有していただきありがとうございます
Kishan Soni 2015

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);

2
しかし、これは廃止されています:(
abbasalim

@ ArMo372、あなたはこれの代わりのコードを見つけましたか?
SimpleCoder 2016

3
ただ、交換するgetBlockSizegetAvailableBlocksしてgetBlockSizeLonggetAvailableBlocksLong
smg 2017

1
これは適切な使用可能スペースを取得していません。@smgが1678ではなく1141になります

1
内部ストレージとしてフォーマットするときにソリューションが機能しない...これを実現する方法を教えてください
Yogesh Rathi

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()使用してではなく、を使用する簡単な方法のように見えますStatFsStatFs@MarkCarter を使用する理由
StuStirling 2014

1
@ DiscoS2 minSdkVersionが9未満の場合はStatFsを使用します
マーク

1
ストレージの変更をどのように監視しますか?
Android開発者

24

使用可能なすべてのストレージフォルダー(SDカードを含む)を取得するには、まずストレージファイルを取得します。

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

次に、それらそれぞれの使用可能なサイズを取得できます。

それには3つの方法があります。

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);

最初のストレージはエミュレートされたものであるため、内部ストレージは最初の外部ストレージと同じである可能性があることに注意してください。


編集:Android Q以上でStorageVolumeを使用すると、次のようなものを使用して、それぞれの空き容量を取得することが可能だと思います:

    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)}")
        }
    }

これが正しいかどうかわからないので、それぞれの合計サイズを取得する方法が見つからないため、ここに書き、ここで質問しまし


1
API 23を搭載したデバイスのリムーバブルSDカード(またはUSB OTGフラッシュドライブ)の空き容量を取得するにはどうすればよいですか?新しいStatFs(file.getPath())。getAvailableBytes()またはfile.getUsableSpace()は、Nexus 5(Marshmallow 6.0.1)の実際のストレージサイズに関係なく、972546048バイトを提供します。
16

@isabsent Nexus 5にはSDカードスロットがありません。どうやってチェックしたの?
Android開発者

USB OTGフラッシュドライブで確認しました。
16

@isabsent使ったことがない。ごめんなさい。API 22以下でうまく機能しますか?
android開発者

1
@Smeet Android 6以上で試すことはできますか?もしそうなら、それはおそらくこのような問題です:code.google.com/p/android/issues/detail?id
android developer

9

@ Android-Droid-あなたはEnvironment.getExternalStorageDirectory()SDカードである必要のない外部ストレージへの間違ったポイントです、それは内部メモリのマウントでもありえます。見る:

外部SDカードの場所を見つける


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のようです。
Acuna

内部ストレージとしてフォーマットすると解決策が機能しない...私を喜ばせてくれますか、これを達成する方法
Yogesh Rathi

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 2016年

内部ストレージとしてフォーマットするときにソリューションが機能しない...これを実現する方法を教えてください
Yogesh Rathi

4

外部メモリトピックへのクイック追加

externalMemoryAvailable()Dinesh Prajapatiの回答のメソッド名と混同しないでください。

Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())メディアが存在し、読み取り/書き込みアクセスでそのマウントポイントにマウントされている場合、メモリの現在の状態を示します。あなたはなりますtrueネクサス5のように、無SD-カードを持つデバイスにも、しかし、まだそれはストレージとの任意の操作の前に「持っている必要があります」メソッドです。

お使いのデバイスに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

別のソリューションをチェックした後、自分でコードを記述してくださいこれは見つけるための完全なコードです

  • 総外部メモリ
  • 無料の外部メモリ
  • 使用された外部メモリ
  • TotaL内部メモリ
  • 使用済み内部メモリ
  • 空き内部メモリ

「」

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)
        }
    }

}}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.