リムーバブルSDカードの場所を見つける


204

外付けSDカードの場所を見つける一般的な方法はありますか?

外部ストレージと混同しないでください。

Environment.getExternalStorageState()「/ mnt / sdcard」のような内部SDマウントポイントへのパスを返します。しかし、問題は外部SDについてです。「/ mnt / sdcard / external_sd」のようなパスを取得するにはどうすればよいですか(デバイスによって異なる場合があります)?

mountファイルシステム名によるコマンドの出力のフィルタリングで終わると思います。しかし、この方法が十分に堅牢であるかどうかはわかりません。


ここではヌガーまで働く私の解決策だ:stackoverflow.com/a/40205116/5002496
Gokul NC

'Environment.getExternalStorageState()は、「/ mnt / sdcard」のような内部SDマウントポイントへのパスを返します。' まあ、Androidがこの用語を使用しているという意味では、内部的なものではありません。お探しの用語は「削除不可能」だと思います。
LarsH

回答:


162

Environment.getExternalStorageState() "/ mnt / sdcard"のような内部SDマウントポイントへのパスを返します

いいえ、Environment.getExternalStorageDirectory()デバイスの製造元が「外部ストレージ」と見なしたものを指します。一部のデバイスでは、これはSDカードなどのリムーバブルメディアです。一部のデバイスでは、これはデバイス上のフラッシュの一部です。ここで、「外部ストレージ」とは、少なくともAndroid 1.xおよび2.xの場合、「ホストマシンにマウントされたときにUSBマスストレージモードを介してアクセスできるもの」を意味します。

しかし、問題は外部SDについてです。「/ mnt / sdcard / external_sd」のようなパスを取得する方法(デバイスごとに異なる場合があります)?

Androidには、上記のように、外部ストレージ以外に「外部SD」の概念はありません。

デバイスの製造元がオンボードフラッシュをオンボードフラッシュに設定し、SDカードも持っている場合は、その製造元に連絡して、SDカードを使用できるかどうか(保証されていません)とルールについて確認する必要があります。使用するパスなど、それを使用する。


更新

最近注目すべき2つのこと:

まず、Android 4.4以降では、リムーバブルメディア(「外部SD」など)への書き込みアクセス権がありません。ただし、getExternalFilesDirs()およびによって返される可能性があるそのメディア上の場所を除きgetExternalCacheDirs()ます。デイブ・スミスの優れた分析をご覧くださいあなたは低レベルの詳細をしたい場合は特に、これのを。

次に、リムーバブルメディアへのアクセスがAndroid SDKの一部であるかどうかについて誰もが誤解しないように、ここにダイアンハックボーンの評価があります

...留意点:Android 4.4まで、公式のAndroidプラットフォームは、2つの特別な場合を除いて、SDカードをまったくサポートしていません:外部ストレージがSDカードである旧式のストレージレイアウト(現在もプラットフォームでサポートされています) 、およびAndroid 3.0に追加された小さな機能で、追加のSDカードをスキャンしてメディアプロバイダーに追加し、ファイルへの読み取り専用アクセスをアプリに付与します(これは現在もプラットフォームでサポートされています)。

Android 4.4は、アプリケーションがストレージにSDカードを使用することを実際に許可したプラットフォームの最初のリリースです。それ以前は、サポートされていないプライベートAPIを介してアクセスしていました。現在、プラットフォームには非常に豊富なAPIがあり、これにより、アプリケーションはサポートされた方法で、これまでよりも優れた方法でSDカードを利用できます。アプリ固有のストレージ領域を自由に使用できます。アプリ内の権限。ファイルピッカーを通過する限り、SDカード上の他のファイルにアクセスできます。特別な権限は必要ありません。


4
そして、この問題は、HCおよびICSデバイスがそのポイント「ExternalStorageDirectory」およびその他のすべてのものが内部の物理ストレージに出てくるにつれて、より大きな問題になっています。さらに、ほとんどのユーザーは、SDカードがファイルシステム内のどこにあるのかを特定する方法がまったくありません。
トニー・マロ

283
だからあなたの答えは基本的に「メーカーに連絡する」です。役に立たない。
dragonroot

6
回答の最後の部分はあまり正確ではありません。実際には、この下の回答(/proc/mounts、/system/etc/vold.fstabなどをスキャンする)に従ってSDカードのパスを検出することは可能です。
OpenGL ESを学ぶ

8
@CommonsWare:それでも、多くのデバイスで機能するソリューションがある場合に製造元に連絡する必要があることは依然として正確ではありません。また、SDK自体がすべてのデバイスで一貫していないため、保証はありません。これらのソリューションがすべてのデバイスで機能しない場合でも、市場に出回っている多くのAndroidアプリが外部SDカードのパスを検出するためにとりわけこれらの技術に依存するのに十分なデバイスで機能します。これらすべての開発者をばかだと呼ぶのは少し厳しく、時期尚早だと思います-確かに顧客はそれを最終的に判断しませんか?
OpenGL ESを学ぶ2013

5
@CommonsWare物事が進むにつれ、それは十分に公平です。開発者がこれがいつでもどこでも機能することを想定できないこと、およびそのようなコードがすべてのデバイスまたはすべてのバージョンのAndroidで機能することを保証できないことは、あなたに間違いなく同意します。うまくいけば、SDKで修正されます。それまでの間、多くのデバイスで機能し、エンドユーザーエクスペリエンスを向上させることができるオプションがまだあります。80%成功と0%成功の間で選択した場合、80%を選択します。
OpenGL ESを

64

ここで見つかったいくつかの回答に基づいて、次の解決策を思いつきました。

コード:

public class ExternalStorage {

    public static final String SD_CARD = "sdCard";
    public static final String EXTERNAL_SD_CARD = "externalSdCard";

    /**
     * @return True if the external storage is available. False otherwise.
     */
    public static boolean isAvailable() {
        String state = Environment.getExternalStorageState();
        if (Environment.MEDIA_MOUNTED.equals(state) || Environment.MEDIA_MOUNTED_READ_ONLY.equals(state)) {
            return true;
        }
        return false;
    }

    public static String getSdCardPath() {
        return Environment.getExternalStorageDirectory().getPath() + "/";
    }

    /**
     * @return True if the external storage is writable. False otherwise.
     */
    public static boolean isWritable() {
        String state = Environment.getExternalStorageState();
        if (Environment.MEDIA_MOUNTED.equals(state)) {
            return true;
        }
        return false;

    }

    /**
     * @return A map of all storage locations available
     */
    public static Map<String, File> getAllStorageLocations() {
        Map<String, File> map = new HashMap<String, File>(10);

        List<String> mMounts = new ArrayList<String>(10);
        List<String> mVold = new ArrayList<String>(10);
        mMounts.add("/mnt/sdcard");
        mVold.add("/mnt/sdcard");

        try {
            File mountFile = new File("/proc/mounts");
            if(mountFile.exists()){
                Scanner scanner = new Scanner(mountFile);
                while (scanner.hasNext()) {
                    String line = scanner.nextLine();
                    if (line.startsWith("/dev/block/vold/")) {
                        String[] lineElements = line.split(" ");
                        String element = lineElements[1];

                        // don't add the default mount path
                        // it's already in the list.
                        if (!element.equals("/mnt/sdcard"))
                            mMounts.add(element);
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        try {
            File voldFile = new File("/system/etc/vold.fstab");
            if(voldFile.exists()){
                Scanner scanner = new Scanner(voldFile);
                while (scanner.hasNext()) {
                    String line = scanner.nextLine();
                    if (line.startsWith("dev_mount")) {
                        String[] lineElements = line.split(" ");
                        String element = lineElements[2];

                        if (element.contains(":"))
                            element = element.substring(0, element.indexOf(":"));
                        if (!element.equals("/mnt/sdcard"))
                            mVold.add(element);
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }


        for (int i = 0; i < mMounts.size(); i++) {
            String mount = mMounts.get(i);
            if (!mVold.contains(mount))
                mMounts.remove(i--);
        }
        mVold.clear();

        List<String> mountHash = new ArrayList<String>(10);

        for(String mount : mMounts){
            File root = new File(mount);
            if (root.exists() && root.isDirectory() && root.canWrite()) {
                File[] list = root.listFiles();
                String hash = "[";
                if(list!=null){
                    for(File f : list){
                        hash += f.getName().hashCode()+":"+f.length()+", ";
                    }
                }
                hash += "]";
                if(!mountHash.contains(hash)){
                    String key = SD_CARD + "_" + map.size();
                    if (map.size() == 0) {
                        key = SD_CARD;
                    } else if (map.size() == 1) {
                        key = EXTERNAL_SD_CARD;
                    }
                    mountHash.add(hash);
                    map.put(key, root);
                }
            }
        }

        mMounts.clear();

        if(map.isEmpty()){
                 map.put(SD_CARD, Environment.getExternalStorageDirectory());
        }
        return map;
    }
}

使用法:

Map<String, File> externalLocations = ExternalStorage.getAllStorageLocations();
File sdCard = externalLocations.get(ExternalStorage.SD_CARD);
File externalSdCard = externalLocations.get(ExternalStorage.EXTERNAL_SD_CARD);

1
ネクサス4、ネクサスs、ギャラクシーs2、ギャラクシーs3、htc desire =)でテスト済み
Richard

2
こんにちは、リチャード-信じられないかもしれませんが、私は尋ねる必要があります。実際にファイルを書き込んだり、この方法でファイルを読み直したりしましたか?以前の「/ sdcard0」の問題を思い出してください。私はこのコードを試しましたが、書き込んだファイルを読み戻そうとしたときにS3 で失敗しました。...これは非常に奇妙です...そして痛いです:))
ハワード・パウツ

10
これは、2つのSDカードを搭載していないデバイスでは失敗します。最初に見つかったものが内部で、2番目に見つかったものが外部であると想定しています...
Caner

Nexus 5とNexus 7のOTGケーブルで接続されたUSBデバイスでは機能しませんでした
Khawar Raza

4
/system/etc/vold.fstabはAndroid 4.3以降ではアクセスできません
Ali

37

ListPreferenceユーザーが何かを保存したい場所の場所を選択する必要がある場所を使用するアプリケーションがありました。

そのアプリでは、SDカードのマウントポイントをスキャン/proc/mounts/system/etc/vold.fstabました。各ファイルのマウントポイントを2つの別々のファイルに保存しましたArrayListの。

次に、1つのリストを他のリストと比較し、両方のリストにないアイテムを破棄しました。これにより、各SDカードへのルートパスのリストが表示されました。

そこから、私はパスをテストしFile.exists()File.isDirectory()、とFile.canWrite()。これらのテストのいずれかが偽だった場合、そのパスをリストから破棄しました。

リストに残っていたものは何でもString[]ListPreferencevalues属性で使用できるように配列に変換しました。

ここにコードを表示できます:http : //sapienmobile.com/?p=204


ちなみに、これはGalaxy S3、SDカード2枚では機能せず、vold.confにリストされているカードは1枚のみです
3c71

1
@ 3c71-Galaxy S3のvoldファイルとマウントファイルを送ってくれませんか?それをカバーするためにコードを微調整します。
男爵

Galaxy S、見つかったすべてのパスは書き込み可能ではなく、奇妙でした。デフォルトの/ mnt / sdcardと/ storage / sdcard0の2つのストレージが見つかりました。どちらもテストに失敗しました
5:

1
マウントファイルを無視するようにコードを調整しました。これは、MotorolaおよびSamsungデバイスの問題でした。マウントファイルは、external_sdのケースをカバーしていませんでしたが、voldにリストされています。私のクラスの最初のバージョンでは、マウントをvoldと比較し、両方に共通ではないアイテムを破棄しました。上記と同じリンクから更新されたクラスを取得します。
2013年

1
バロンに感謝します。これが「その」答えです。少なくとも有用なもの。
pstoppani 2013

23

ContextCompat.getExternalFilesDirs()のサポートライブラリ関数を使用してみてください。

      final File[] appsDir=ContextCompat.getExternalFilesDirs(getActivity(),null);
      final ArrayList<File> extRootPaths=new ArrayList<>();
      for(final File file : appsDir)
        extRootPaths.add(file.getParentFile().getParentFile().getParentFile().getParentFile());

1つ目はプライマリ外部ストレージで、残りは実際のSDカードパスです。

「.getParentFile()」が複数ある理由は、元のパスが

.../Android/data/YOUR_APP_PACKAGE_NAME/files/

編集:SDカードのパスを取得するために私が作成したより包括的な方法は次のとおりです:

  /**
   * returns a list of all available sd cards paths, or null if not found.
   *
   * @param includePrimaryExternalStorage set to true if you wish to also include the path of the primary external storage
   */
  @TargetApi(Build.VERSION_CODES.HONEYCOMB)
  public static List<String> getSdCardPaths(final Context context, final boolean includePrimaryExternalStorage)
    {
    final File[] externalCacheDirs=ContextCompat.getExternalCacheDirs(context);
    if(externalCacheDirs==null||externalCacheDirs.length==0)
      return null;
    if(externalCacheDirs.length==1)
      {
      if(externalCacheDirs[0]==null)
        return null;
      final String storageState=EnvironmentCompat.getStorageState(externalCacheDirs[0]);
      if(!Environment.MEDIA_MOUNTED.equals(storageState))
        return null;
      if(!includePrimaryExternalStorage&&VERSION.SDK_INT>=VERSION_CODES.HONEYCOMB&&Environment.isExternalStorageEmulated())
        return null;
      }
    final List<String> result=new ArrayList<>();
    if(includePrimaryExternalStorage||externalCacheDirs.length==1)
      result.add(getRootOfInnerSdCardFolder(externalCacheDirs[0]));
    for(int i=1;i<externalCacheDirs.length;++i)
      {
      final File file=externalCacheDirs[i];
      if(file==null)
        continue;
      final String storageState=EnvironmentCompat.getStorageState(file);
      if(Environment.MEDIA_MOUNTED.equals(storageState))
        result.add(getRootOfInnerSdCardFolder(externalCacheDirs[i]));
      }
    if(result.isEmpty())
      return null;
    return result;
    }

  /** Given any file/folder inside an sd card, this will return the path of the sd card */
  private static String getRootOfInnerSdCardFolder(File file)
    {
    if(file==null)
      return null;
    final long totalSpace=file.getTotalSpace();
    while(true)
      {
      final File parentFile=file.getParentFile();
      if(parentFile==null||parentFile.getTotalSpace()!=totalSpace||!parentFile.canRead())
        return file.getAbsolutePath();
      file=parentFile;
      }
    }

非常に良い答えのようですが、これを単純なアクティビティにどのように統合しますか?定義されていない複数の変数などがありAppContextCompactEnvironmentCompact
アントニオ・

@Antonio ContextCompact、EnvironmentCompactは、サポートライブラリから利用できます。「App.global()」は単なるアプリケーションコンテキストであり、どこにでもContextパラメータを追加したくないので、グローバルに設定しています。
Android開発者

1
すごい!私のデバイスv4.4 Samsung GT S Advanceで動作します。他の
ユーザーで

@androiddeveloper編集した回答はすべてのデバイスとSDカードサイズで機能しますか?
Rahulrr2602 2017

1
これは私にとって完璧に機能しました-受け入れられる答えになるはずです。
Paradox

17

すべての外部ストレージを取得するには(SDカードであるか、リムーバブルでない内部ストレージであるかに関係なく)、次のコードを使用できます。

final String state = Environment.getExternalStorageState();

if ( Environment.MEDIA_MOUNTED.equals(state) || Environment.MEDIA_MOUNTED_READ_ONLY.equals(state) ) {  // we can read the External Storage...           
    //Retrieve the primary External Storage:
    final File primaryExternalStorage = Environment.getExternalStorageDirectory();

    //Retrieve the External Storages root directory:
    final String externalStorageRootDir;
    if ( (externalStorageRootDir = primaryExternalStorage.getParent()) == null ) {  // no parent...
        Log.d(TAG, "External Storage: " + primaryExternalStorage + "\n");
    }
    else {
        final File externalStorageRoot = new File( externalStorageRootDir );
        final File[] files = externalStorageRoot.listFiles();

        for ( final File file : files ) {
            if ( file.isDirectory() && file.canRead() && (file.listFiles().length > 0) ) {  // it is a real directory (not a USB drive)...
                Log.d(TAG, "External Storage: " + file.getAbsolutePath() + "\n");
            }
        }
    }
}

または、System.getenv( "EXTERNAL_STORAGE")を使用してプライマリ外部ストレージディレクトリ(例:"/ storage / sdcard0")を取得しSystem.getenv( "SECONDARY_STORAGE")を使用してすべてのセカンダリディレクトリ(例:" / storage / extSdCard:/ storage / UsbDriveA:/ storage / UsbDriveB ")。この場合も、USBドライブを除外するために、セカンダリディレクトリのリストをフィルタリングすることをお勧めします。

いずれにせよ、ハードコードされたパスを使用することは常に悪いアプローチであることに注意してください(特に、すべての製造元が満足できるように変更する場合)


2
コメントを残さないダウン投票者はトロールだと考えてください。;)しかし、あなたの方法はかなり恣意的だと​​思います。質問で尋ねられたように、これらの「USBドライブ」をスキップし、他のすべてを維持することが「SDカード」に等しいことをどうやって知ることができますか?また、System.getenv("SECONDARY_STORAGE")文書化されていないように見えるため、提案はいくつかの参照でも実行できます。
Sz。

1
私の知る限り、Android APIでは、すべての外部ストレージを取得するための標準メソッドの参照はありません。しかしながら、提案された方法は全く恣意的ではない。Androidでは、すべてのUnix / Linuxシステムと同様に、すべてのマウントストレージデバイスが共通のディレクトリ「/ mnt」(ストレージデバイスをマウントするための標準のUnix / Linuxディレクトリ)または最新バージョンでは「/ストレージ"。このため、このフォルダーにリンクされているすべてのSDカードを見つけることができます。
Paolo Rovelli

1
System.getenv( "EXTERNAL_STORAGE")メソッドに関しては、APIページ(詳細は説明されていません)ではなく参照がありません:developer.android.com/reference/java/lang/…何も見つかりませんでしたAndroidシステム環境変数の公式ページ。ただし、ここにそれらの短いリストを見つけることができます:herongyang.com/Android/…–
Paolo Rovelli

/mntSDカードについて確信が持てないということは、SDカードやUSBドライブだけでなく、他にもさまざまなfsツリーが存在する可能性があるということです。あなたのコードは、私が正しく理解していれば、内部(おそらく仮想)ファイルシステムのマウントもリストしますが、質問ではsdカードのみが必要です
Sz。

1
そうですか。はい、そうです。私の方法では、内部(取り外し不可)SDメモリも取得します。
Paolo Rovelli、2014年

15

Richardと同様に、/ proc / mountsファイルを使用して、使用可能なストレージオプションのリストを取得します

public class StorageUtils {

    private static final String TAG = "StorageUtils";

    public static class StorageInfo {

        public final String path;
        public final boolean internal;
        public final boolean readonly;
        public final int display_number;

        StorageInfo(String path, boolean internal, boolean readonly, int display_number) {
            this.path = path;
            this.internal = internal;
            this.readonly = readonly;
            this.display_number = display_number;
        }

        public String getDisplayName() {
            StringBuilder res = new StringBuilder();
            if (internal) {
                res.append("Internal SD card");
            } else if (display_number > 1) {
                res.append("SD card " + display_number);
            } else {
                res.append("SD card");
            }
            if (readonly) {
                res.append(" (Read only)");
            }
            return res.toString();
        }
    }

    public static List<StorageInfo> getStorageList() {

        List<StorageInfo> list = new ArrayList<StorageInfo>();
        String def_path = Environment.getExternalStorageDirectory().getPath();
        boolean def_path_internal = !Environment.isExternalStorageRemovable();
        String def_path_state = Environment.getExternalStorageState();
        boolean def_path_available = def_path_state.equals(Environment.MEDIA_MOUNTED)
                                    || def_path_state.equals(Environment.MEDIA_MOUNTED_READ_ONLY);
        boolean def_path_readonly = Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED_READ_ONLY);
        BufferedReader buf_reader = null;
        try {
            HashSet<String> paths = new HashSet<String>();
            buf_reader = new BufferedReader(new FileReader("/proc/mounts"));
            String line;
            int cur_display_number = 1;
            Log.d(TAG, "/proc/mounts");
            while ((line = buf_reader.readLine()) != null) {
                Log.d(TAG, line);
                if (line.contains("vfat") || line.contains("/mnt")) {
                    StringTokenizer tokens = new StringTokenizer(line, " ");
                    String unused = tokens.nextToken(); //device
                    String mount_point = tokens.nextToken(); //mount point
                    if (paths.contains(mount_point)) {
                        continue;
                    }
                    unused = tokens.nextToken(); //file system
                    List<String> flags = Arrays.asList(tokens.nextToken().split(",")); //flags
                    boolean readonly = flags.contains("ro");

                    if (mount_point.equals(def_path)) {
                        paths.add(def_path);
                        list.add(0, new StorageInfo(def_path, def_path_internal, readonly, -1));
                    } else if (line.contains("/dev/block/vold")) {
                        if (!line.contains("/mnt/secure")
                            && !line.contains("/mnt/asec")
                            && !line.contains("/mnt/obb")
                            && !line.contains("/dev/mapper")
                            && !line.contains("tmpfs")) {
                            paths.add(mount_point);
                            list.add(new StorageInfo(mount_point, false, readonly, cur_display_number++));
                        }
                    }
                }
            }

            if (!paths.contains(def_path) && def_path_available) {
                list.add(0, new StorageInfo(def_path, def_path_internal, def_path_readonly, -1));
            }

        } catch (FileNotFoundException ex) {
            ex.printStackTrace();
        } catch (IOException ex) {
            ex.printStackTrace();
        } finally {
            if (buf_reader != null) {
                try {
                    buf_reader.close();
                } catch (IOException ex) {}
            }
        }
        return list;
    }    
}

ありがとうございます。完璧に働きました。また、StorageInfoを変更不可にした方法も気に入っています。一方でprintStackTrace?いつandroid.util.Log.e
マーティン

1
Nexus 5とNexus 7のOTGケーブルで接続されたUSBデバイスでは機能しませんでした
Khawar Raza

1
これを使用してSDカードにファイルを書き込むことはできません
Eu Vid

@EuVidと同じ問題がVM / AVDでは機能しますが、ハードウェアでは機能しません
スパイ

11

追加のSDカードがマウントされている場所を見つけるには、/proc/mounts標準のLinuxファイルを読み取り、voldデータとクロスチェックします/system/etc/vold.conf)。また、によって返される場所はEnvironment.getExternalStorageDirectory()vold構成には表示されない場合があります(一部のデバイスでは、マウント解除できない内部ストレージです)が、それでもリストに含める必要があります。しかし、ユーザーにそれらを説明する良い方法が見つかりませんでした。


Imo、ファイルシステムのmount読み取りよりも使用の方が互換性があり/procます。問題は、SDカードがFATとしてフォーマットされている必要がないことです。また、カードのマウントポイントはROMによって異なります。また、他にいくつかのVFATパーティションが存在する可能性があります...
borisstr

1
@borisstr:うーん、実際にはAndroidはvoldを使用しているので、設定も確認するのが適切です。
Jan Hudec、

上記の投稿から共有したコードファイルには、検出されたルートパスをユーザーに説明する方法が含まれています。setProperties()メソッドを見てください。
男爵

1
@borisstr、実際には、いいえ。/proc/mountsの読み取りは、mount実行可能ファイルを起動するよりもAndroidデバイスで移植性が高くなります。特に、実行可能ファイルの起動は推奨されません。
Chris Stratton、

7

今回はこのトピック内のすべてのソリューションを試します。しかし、それらすべては、1つの外部(取り外し可能)および1つの内部(取り外し不可能)カードを備えたデバイスでは正しく機能しませんでした。外部カードのパスは「mount」コマンド、「proc / mounts」ファイルなどから取得できません

そして、私は自分のソリューションを作成します(Paulo Luanの場合):

String sSDpath = null;
File   fileCur = null;
for( String sPathCur : Arrays.asList( "ext_card", "external_sd", "ext_sd", "external", "extSdCard",  "externalSdCard")) // external sdcard
{
   fileCur = new File( "/mnt/", sPathCur);
   if( fileCur.isDirectory() && fileCur.canWrite())
   {
     sSDpath = fileCur.getAbsolutePath();
     break;
   }
}
fileCur = null;
if( sSDpath == null)  sSDpath = Environment.getExternalStorageDirectory().getAbsolutePath();

6

ソースコードをandroid.os.Environment見ると、Androidがパスの環境変数に大きく依存していることがわかります。「SECONDARY_STORAGE」環境変数を使用して、リムーバブルSDカードへのパスを見つけることができます。

/**
 * Get a file using an environmental variable.
 *
 * @param variableName
 *         The Environment variable name.
 * @param paths
 *         Any paths to the file if the Environment variable was not found.
 * @return the File or {@code null} if the File could not be located.
 */
private static File getDirectory(String variableName, String... paths) {
    String path = System.getenv(variableName);
    if (!TextUtils.isEmpty(path)) {
        if (path.contains(":")) {
            for (String _path : path.split(":")) {
                File file = new File(_path);
                if (file.exists()) {
                    return file;
                }
            }
        } else {
            File file = new File(path);
            if (file.exists()) {
                return file;
            }
        }
    }
    if (paths != null && paths.length > 0) {
        for (String _path : paths) {
            File file = new File(_path);
            if (file.exists()) {
                return file;
            }
        }
    }
    return null;
}

使用例:

public static final File REMOVABLE_STORAGE = getDirectory("SECONDARY_STORAGE");

5

単にこれを使用してください:

String primary_sd = System.getenv("EXTERNAL_STORAGE");
if(primary_sd != null)
    Log.i("EXTERNAL_STORAGE", primary_sd);
String secondary_sd = System.getenv("SECONDARY_STORAGE");
if(secondary_sd != null)
    Log.i("SECONDARY_STORAGE", secondary_sd)

一部のデバイスでSECONDARY_STORAGEは、コロン( ":")で区切られた複数のパスがあります。これが、文字列を分割する理由です(上記の私の回答を参照してください)。
Jared Rummler、2015年

これらはどちらもnullを返します。
Tim Cooper、

5

外付けSDカードの場所を見つける一般的な方法はありますか?

ことで普遍的な方法、あなたは公式の方法を意味する場合は、はい、あります。

APIレベル19、つまりAndroidバージョン4.4 KitkatではFile[] getExternalFilesDirs (String type)ContextアプリがデータやファイルをマイクロSDカードに保存できるようにするクラスにています。

Android 4.4は、アプリがストレージにSDカードを使用することを実際に許可したプラットフォームの最初のリリースです。APIレベル19より前のSDカードへのアクセスは、サポートされていないプライベートAPIを介して行われていました。

getExternalFilesDirs(String type)は、すべての共有/外部ストレージデバイス上のアプリケーション固有のディレクトリへの絶対パスを返します。つまり、内部メモリと外部メモリの両方へのパスを返します。通常、2番目に返されるパスは、microSDカード(存在する場合)のストレージパスです。

ただし、

リムーバブルメディアはユーザーが取り出すことができるため、共有ストレージは常に利用できるとは限りません。メディアの状態は、を使用して確認できます getExternalStorageState(File)

これらのファイルに適用されるセキュリティはありません。たとえば、保持しているアプリケーションWRITE_EXTERNAL_STORAGEはこれらのファイルに書き込むことができます。

Google /公式Androidドキュメントによる内部および外部ストレージの用語は、私たちの考えとはかなり異なります。


「Google /公式Androidドキュメントによる内部および外部ストレージの用語は、私たちの考えとはかなり異なります。」はい、実際、質問のタイトルは、OPが取り外し可能な SDカードについて尋ねていることを明確に示しています。getExternalFilesDirs()はリムーバブルではないSDカードを返すことが多いため、これはリムーバブルSDカードの場所を見つける一般的な方法ではありません。
LarsH

「getExternalFilesDirs(String type)は、すべての共有/外部ストレージデバイス上のアプリケーション固有のディレクトリへの絶対パスを返します。つまり、内部メモリと外部メモリの両方へのパスを返します。」このペアの文は非常に誤解を招くものです。なぜなら、両方が真であるためには、「外部」は2つの異なる矛盾することを意味する必要があるためです。
LarsH

4

これが、外部カードを見つけるために使用する方法です。mount cmd returnを使用してから、vfat部分を解析します。

String s = "";
try {
Process process = new ProcessBuilder().command("mount")
        .redirectErrorStream(true).start();

process.waitFor();

InputStream is = process.getInputStream();
byte[] buffer = new byte[1024];
while (is.read(buffer) != -1) {
    s = s + new String(buffer);
}
is.close();
} catch (Exception e) {
e.printStackTrace();
}

//用行分隔mount列表
String[] lines = s.split("\n");
for(int i=0; i<lines.length; i++) {
//如果行内有挂载路径且为vfat类型,说明可能是内置或者外置sd的挂载点
if(-1 != lines[i].indexOf(path[0]) && -1 != lines[i].indexOf("vfat")) {
    //再用空格分隔
    String[] blocks = lines[i].split("\\s");
    for(int j=0; j<blocks.length; j++) {
        //判断是否是挂载为vfat类型
        if(-1 != blocks[j].indexOf(path[0])) {
            //Test if it is the external sd card.
        }
    }
}
}

4

このソリューションは、 System.getenv("SECONDARY_STORAGE")マシュマロでは役に立たないます。

テスト済みで作業中:

  • Samsung Galaxy Tab 2(Android 4.1.1-Stock)
  • Samsung Galaxy Note 8.0(Android 4.2.2-Stock)
  • Samsung Galaxy S4(Android 4.4-株式)
  • Samsung Galaxy S4(Android 5.1.1-Cyanogenmod)
  • Samsung Galaxy Tab A(Android 6.0.1-Stock)

    /**
     * Returns all available external SD-Card roots in the system.
     *
     * @return paths to all available external SD-Card roots in the system.
     */
    public static String[] getStorageDirectories() {
        String [] storageDirectories;
        String rawSecondaryStoragesStr = System.getenv("SECONDARY_STORAGE");
    
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            List<String> results = new ArrayList<String>();
            File[] externalDirs = applicationContext.getExternalFilesDirs(null);
            for (File file : externalDirs) {
                String path = file.getPath().split("/Android")[0];
                if((Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP && Environment.isExternalStorageRemovable(file))
                        || rawSecondaryStoragesStr != null && rawSecondaryStoragesStr.contains(path)){
                    results.add(path);
                }
            }
            storageDirectories = results.toArray(new String[0]);
        }else{
            final Set<String> rv = new HashSet<String>();
    
            if (!TextUtils.isEmpty(rawSecondaryStoragesStr)) {
                final String[] rawSecondaryStorages = rawSecondaryStoragesStr.split(File.pathSeparator);
                Collections.addAll(rv, rawSecondaryStorages);
            }
            storageDirectories = rv.toArray(new String[rv.size()]);
        }
        return storageDirectories;
    }

2

上記の私の最初の回答以来、voldをスキャンすることは、さまざまなメーカーで実行可能ではありません。

私はより信頼性が高く、簡単な方法を開発しました。

File mnt = new File("/storage");
if (!mnt.exists())
    mnt = new File("/mnt");

File[] roots = mnt.listFiles(new FileFilter() {

    @Override
    public boolean accept(File pathname) {
        return pathname.isDirectory() && pathname.exists()
                && pathname.canWrite() && !pathname.isHidden()
                && !isSymlink(pathname);
    }
});

ルートには、USB接続されたUSBデバイスを含む、システム上のすべての書き込み可能なルートディレクトリが含まれます。

注:canWriteメソッドには、android.permission.WRITE_EXTERNAL_STORAGE権限が必要です。


メソッドisSymlink(File)は、タイプnew FileFilter()に対して未定義です{}
Omid

canWriteが原因でこれがAndroid 4.4で外部sdカードを見つけられない場合のアイデアはありますか?
アンソニー

これは確かに他の方法より簡単ですが、信頼できますか?たとえば、サムスンの一部のデバイスで/external_sdは、外部microSDカードであると読みました。一部のLGでは、そう/_ExternalSDです。一部のデバイスでは/sdcardです。後者はおそらくシンボリックリンク/storage/sdcard0またはそれに似ていますが、これらの他のものは本当に確実に/storage/*and によってカバーされ/mount/*ますか?
LarsH

また、pathname.canWrite()WRITE_EXTERNAL_STORAGE権限を使用する必要がありますか?なぜ電話しないのpathname.canRead()ですか?
LarsH

1

それはとても遅いですが、ついに私はほとんどのデバイス(メーカーとAndroidバージョンによって)をテストした何かを得ました、それはAndroid 2.2+で動作しています。機能していない場合は、デバイス名でコメントしてください。私はそれを修正します。誰か興味があれば私はそれがどのように機能するかを説明します。

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FilenameFilter;
import java.io.IOException;
import java.io.InputStreamReader;

import android.util.Log;


/**
 * @author ajeet
 *05-Dec-2014  2014
 *
 */
public class StorageUtil {

    public boolean isRemovebleSDCardMounted() {
        File file = new File("/sys/class/block/");
        File[] files = file.listFiles(new MmcblkFilter("mmcblk\\d$"));
        boolean flag = false;
        for (File mmcfile : files) {
            File scrfile = new File(mmcfile, "device/scr");
            if (scrfile.exists()) {
                flag = true;
                break;
            }
        }
        return flag;
    }

    public String getRemovebleSDCardPath() throws IOException {
        String sdpath = null;
        File file = new File("/sys/class/block/");
        File[] files = file.listFiles(new MmcblkFilter("mmcblk\\d$"));
        String sdcardDevfile = null;
        for (File mmcfile : files) {
            Log.d("SDCARD", mmcfile.getAbsolutePath());
            File scrfile = new File(mmcfile, "device/scr");
            if (scrfile.exists()) {
                sdcardDevfile = mmcfile.getName();
                Log.d("SDCARD", mmcfile.getName());
                break;
            }
        }
        if (sdcardDevfile == null) {
            return null;
        }
        FileInputStream is;
        BufferedReader reader;

        files = file.listFiles(new MmcblkFilter(sdcardDevfile + "p\\d+"));
        String deviceName = null;
        if (files.length > 0) {
            Log.d("SDCARD", files[0].getAbsolutePath());
            File devfile = new File(files[0], "dev");
            if (devfile.exists()) {
                FileInputStream fis = new FileInputStream(devfile);
                reader = new BufferedReader(new InputStreamReader(fis));
                String line = reader.readLine();
                deviceName = line;
            }
            Log.d("SDCARD", "" + deviceName);
            if (deviceName == null) {
                return null;
            }
            Log.d("SDCARD", deviceName);

            final File mountFile = new File("/proc/self/mountinfo");

            if (mountFile.exists()) {
                is = new FileInputStream(mountFile);
                reader = new BufferedReader(new InputStreamReader(is));
                String line = null;
                while ((line = reader.readLine()) != null) {
                    // Log.d("SDCARD", line);
                    // line = reader.readLine();
                    // Log.d("SDCARD", line);
                    String[] mPonts = line.split("\\s+");
                    if (mPonts.length > 6) {
                        if (mPonts[2].trim().equalsIgnoreCase(deviceName)) {
                            if (mPonts[4].contains(".android_secure")
                                    || mPonts[4].contains("asec")) {
                                continue;
                            }
                            sdpath = mPonts[4];
                            Log.d("SDCARD", mPonts[4]);

                        }
                    }

                }
            }

        }

        return sdpath;
    }

    static class MmcblkFilter implements FilenameFilter {
        private String pattern;

        public MmcblkFilter(String pattern) {
            this.pattern = pattern;

        }

        @Override
        public boolean accept(File dir, String filename) {
            if (filename.matches(pattern)) {
                return true;
            }
            return false;
        }

    }

}

ダウンボッターねえ、まず試してみてください。それが機能しない場合は、デバイスにコメントしてください。私たちは、Android 2.2以降を搭載した何千ものデバイスでそれを使用しています
Ajeet47

あなたのクラスは私にSamsung Galaxy S4、GT-I9500、Android 5.0.1(デバイスがルート化されていない)で/ mnt / media_rw / extSdCardを与えます。しかし、ESファイルマネージャを使用してフォルダ/ mnt / media_rwに表示されるものはありません...
isabsent

@isabsent use if(Build.VERSION.SDK_INT> = Build.VERSION_CODES.KITKAT){File [] file = context.getExternalFilesDirs(null); return file.length> 1?file [1]:null; }
Ajeet47 2015

stackoverflow.com/a/27197248/753575についてどう思いますか?Build.VERSION.SDK_INT> = Build.VERSION_CODES.KITKATの場合、このアプローチはより包括的ですか?
2015

はい。context.getExternalFilesDirs(null)から意味のあるパスを取得することが保証されていますが、ルートパスに対してそれをトリムします(それはアプリディレクトリのパスを返します。 "/ Android"でそれをトリムします)
Ajeet47

1

以下のコードを記述することで、場所を取得します。

/ storage / 663D-554E / Android / data / app_package_name / files /

アプリのデータをsd_card内の/ android / dataの場所に保存します。

File[] list = ContextCompat.getExternalFilesDirs(MainActivity.this, null);

list[1]+"/fol" 

場所を取得するには、内部に0、SDカードに1をファイル配列に渡します。

私はこのコードをmoto g4 plusおよびSamsungデバイスでテストしました(すべて正常に動作します)。

これがお役に立てば幸いです。


sdカードのパスがインデックス1にない場合があります。インデックス0にある場合があります。他の方法に従う方がよい
Raghav Satyadev

1

これがリムーバブルを見つけるために使用する方法です SDカードです。複雑で、状況によってはやり過ぎかもしれませんが、過去数年間テストしたAndroidのさまざまなバージョンやデバイスメーカーで動作します。マウントされているSDカードが見つからないAPIレベル15以降のデバイスについては知りません。検索する既知のファイルの名前を指定した場合は特に、ほとんどの場合、誤検知は返されません。

うまくいかない場合はお知らせください。

import android.content.Context;
import android.os.Build;
import android.os.Environment;
import android.support.v4.content.ContextCompat;
import android.text.TextUtils;
import android.util.Log;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.Locale;
import java.util.regex.Pattern;

public class SDCard {
    private static final String TAG = "SDCard";

    /** In some scenarios we can expect to find a specified file or folder on SD cards designed
     * to work with this app. If so, set KNOWNFILE to that filename. It will make our job easier.
     * Set it to null otherwise. */
    private static final String KNOWNFILE = null;

    /** Common paths for microSD card. **/
    private static String[] commonPaths = {
            // Some of these taken from
            // /programming/13976982/removable-storage-external-sdcard-path-by-manufacturers
            // These are roughly in order such that the earlier ones, if they exist, are more sure
            // to be removable storage than the later ones.
            "/mnt/Removable/MicroSD",
            "/storage/removable/sdcard1", // !< Sony Xperia Z1
            "/Removable/MicroSD", // Asus ZenPad C
            "/removable/microsd",
            "/external_sd", // Samsung
            "/_ExternalSD", // some LGs
            "/storage/extSdCard", // later Samsung
            "/storage/extsdcard", // Main filesystem is case-sensitive; FAT isn't.
            "/mnt/extsd", // some Chinese tablets, e.g. Zeki
            "/storage/sdcard1", // If this exists it's more likely than sdcard0 to be removable.
            "/mnt/extSdCard",
            "/mnt/sdcard/external_sd",
            "/mnt/external_sd",
            "/storage/external_SD",
            "/storage/ext_sd", // HTC One Max
            "/mnt/sdcard/_ExternalSD",
            "/mnt/sdcard-ext",

            "/sdcard2", // HTC One M8s
            "/sdcard1", // Sony Xperia Z
            "/mnt/media_rw/sdcard1",   // 4.4.2 on CyanogenMod S3
            "/mnt/sdcard", // This can be built-in storage (non-removable).
            "/sdcard",
            "/storage/sdcard0",
            "/emmc",
            "/mnt/emmc",
            "/sdcard/sd",
            "/mnt/sdcard/bpemmctest",
            "/mnt/external1",
            "/data/sdext4",
            "/data/sdext3",
            "/data/sdext2",
            "/data/sdext",
            "/storage/microsd" //ASUS ZenFone 2

            // If we ever decide to support USB OTG storage, the following paths could be helpful:
            // An LG Nexus 5 apparently uses usb://1002/UsbStorage/ as a URI to access an SD
            // card over OTG cable. Other models, like Galaxy S5, use /storage/UsbDriveA
            //        "/mnt/usb_storage",
            //        "/mnt/UsbDriveA",
            //        "/mnt/UsbDriveB",
    };

    /** Find path to removable SD card. */
    public static File findSdCardPath(Context context) {
        String[] mountFields;
        BufferedReader bufferedReader = null;
        String lineRead = null;

        /** Possible SD card paths */
        LinkedHashSet<File> candidatePaths = new LinkedHashSet<>();

        /** Build a list of candidate paths, roughly in order of preference. That way if
         * we can't definitively detect removable storage, we at least can pick a more likely
         * candidate. */

        // Could do: use getExternalStorageState(File path), with and without an argument, when
        // available. With an argument is available since API level 21.
        // This may not be necessary, since we also check whether a directory exists and has contents,
        // which would fail if the external storage state is neither MOUNTED nor MOUNTED_READ_ONLY.

        // I moved hard-coded paths toward the end, but we need to make sure we put the ones in
        // backwards order that are returned by the OS. And make sure the iterators respect
        // the order!
        // This is because when multiple "external" storage paths are returned, it's always (in
        // experience, but not guaranteed by documentation) with internal/emulated storage
        // first, removable storage second.

        // Add value of environment variables as candidates, if set:
        // EXTERNAL_STORAGE, SECONDARY_STORAGE, EXTERNAL_SDCARD_STORAGE
        // But note they are *not* necessarily *removable* storage! Especially EXTERNAL_STORAGE.
        // And they are not documented (API) features. Typically useful only for old versions of Android.

        String val = System.getenv("SECONDARY_STORAGE");
        if (!TextUtils.isEmpty(val)) addPath(val, null, candidatePaths);
        val = System.getenv("EXTERNAL_SDCARD_STORAGE");
        if (!TextUtils.isEmpty(val)) addPath(val, null, candidatePaths);

        // Get listing of mounted devices with their properties.
        ArrayList<File> mountedPaths = new ArrayList<>();
        try {
            // Note: Despite restricting some access to /proc (http://stackoverflow.com/a/38728738/423105),
            // Android 7.0 does *not* block access to /proc/mounts, according to our test on George's Alcatel A30 GSM.
            bufferedReader = new BufferedReader(new FileReader("/proc/mounts"));

            // Iterate over each line of the mounts listing.
            while ((lineRead = bufferedReader.readLine()) != null) {
                Log.d(TAG, "\nMounts line: " + lineRead);
                mountFields = lineRead.split(" ");

                // columns: device, mountpoint, fs type, options... Example:
                // /dev/block/vold/179:97 /storage/sdcard1 vfat rw,dirsync,nosuid,nodev,noexec,relatime,uid=1000,gid=1015,fmask=0002,dmask=0002,allow_utime=0020,codepage=cp437,iocharset=iso8859-1,shortname=mixed,utf8,errors=remount-ro 0 0
                String device = mountFields[0], path = mountFields[1], fsType = mountFields[2];

                // The device, path, and fs type must conform to expected patterns.
                if (!(devicePattern.matcher(device).matches() &&
                        pathPattern.matcher(path).matches() &&
                        fsTypePattern.matcher(fsType).matches()) ||
                        // mtdblock is internal, I'm told.
                        device.contains("mtdblock") ||
                        // Check for disqualifying patterns in the path.
                        pathAntiPattern.matcher(path).matches()) {
                    // If this mounts line fails our tests, skip it.
                    continue;
                }

                // TODO maybe: check options to make sure it's mounted RW?
                // The answer at http://stackoverflow.com/a/13648873/423105 does.
                // But it hasn't seemed to be necessary so far in my testing.

                // This line met the criteria so far, so add it to candidate list.
                addPath(path, null, mountedPaths);
            }
        } catch (IOException ignored) {
        } finally {
            if (bufferedReader != null) {
                try {
                    bufferedReader.close();
                } catch (IOException ignored) {
                }
            }
        }

        // Append the paths from mount table to candidate list, in reverse order.
        if (!mountedPaths.isEmpty()) {
            // See https://stackoverflow.com/a/5374346/423105 on why the following is necessary.
            // Basically, .toArray() needs its parameter to know what type of array to return.
            File[] mountedPathsArray = mountedPaths.toArray(new File[mountedPaths.size()]);
            addAncestors(candidatePaths, mountedPathsArray);
        }

        // Add hard-coded known common paths to candidate list:
        addStrings(candidatePaths, commonPaths);

        // If the above doesn't work we could try the following other options, but in my experience they
        // haven't added anything helpful yet.

        // getExternalFilesDir() and getExternalStorageDirectory() typically something app-specific like
        //   /storage/sdcard1/Android/data/com.mybackuparchives.android/files
        // so we want the great-great-grandparent folder.

        // This may be non-removable.
        Log.d(TAG, "Environment.getExternalStorageDirectory():");
        addPath(null, ancestor(Environment.getExternalStorageDirectory()), candidatePaths);

        // Context.getExternalFilesDirs() is only available from API level 19. You can use
        // ContextCompat.getExternalFilesDirs() on earlier APIs, but it only returns one dir anyway.
        Log.d(TAG, "context.getExternalFilesDir(null):");
        addPath(null, ancestor(context.getExternalFilesDir(null)), candidatePaths);

        // "Returns absolute paths to application-specific directories on all external storage
        // devices where the application can place persistent files it owns."
        // We might be able to use these to deduce a higher-level folder that isn't app-specific.
        // Also, we apparently have to call getExternalFilesDir[s](), at least in KITKAT+, in order to ensure that the
        // "external files" directory exists and is available.
        Log.d(TAG, "ContextCompat.getExternalFilesDirs(context, null):");
        addAncestors(candidatePaths, ContextCompat.getExternalFilesDirs(context, null));
        // Very similar results:
        Log.d(TAG, "ContextCompat.getExternalCacheDirs(context):");
        addAncestors(candidatePaths, ContextCompat.getExternalCacheDirs(context));

        // TODO maybe: use getExternalStorageState(File path), with and without an argument, when
        // available. With an argument is available since API level 21.
        // This may not be necessary, since we also check whether a directory exists,
        // which would fail if the external storage state is neither MOUNTED nor MOUNTED_READ_ONLY.

        // A "public" external storage directory. But in my experience it doesn't add anything helpful.
        // Note that you can't pass null, or you'll get an NPE.
        final File publicDirectory = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_MUSIC);
        // Take the parent, because we tend to get a path like /pathTo/sdCard/Music.
        addPath(null, publicDirectory.getParentFile(), candidatePaths);
        // EXTERNAL_STORAGE: may not be removable.
        val = System.getenv("EXTERNAL_STORAGE");
        if (!TextUtils.isEmpty(val)) addPath(val, null, candidatePaths);

        if (candidatePaths.isEmpty()) {
            Log.w(TAG, "No removable microSD card found.");
            return null;
        } else {
            Log.i(TAG, "\nFound potential removable storage locations: " + candidatePaths);
        }

        // Accept or eliminate candidate paths if we can determine whether they're removable storage.
        // In Lollipop and later, we can check isExternalStorageRemovable() status on each candidate.
        if (Build.VERSION.SDK_INT >= 21) {
            Iterator<File> itf = candidatePaths.iterator();
            while (itf.hasNext()) {
                File dir = itf.next();
                // handle illegalArgumentException if the path is not a valid storage device.
                try {
                    if (Environment.isExternalStorageRemovable(dir)
                        // && containsKnownFile(dir)
                            ) {
                        Log.i(TAG, dir.getPath() + " is removable external storage");
                        return dir;
                    } else if (Environment.isExternalStorageEmulated(dir)) {
                        Log.d(TAG, "Removing emulated external storage dir " + dir);
                        itf.remove();
                    }
                } catch (IllegalArgumentException e) {
                    Log.d(TAG, "isRemovable(" + dir.getPath() + "): not a valid storage device.", e);
                }
            }
        }

        // Continue trying to accept or eliminate candidate paths based on whether they're removable storage.
        // On pre-Lollipop, we only have singular externalStorage. Check whether it's removable.
        if (Build.VERSION.SDK_INT >= 9) {
            File externalStorage = Environment.getExternalStorageDirectory();
            Log.d(TAG, String.format(Locale.ROOT, "findSDCardPath: getExternalStorageDirectory = %s", externalStorage.getPath()));
            if (Environment.isExternalStorageRemovable()) {
                // Make sure this is a candidate.
                // TODO: Does this contains() work? Should we be canonicalizing paths before comparing?
                if (candidatePaths.contains(externalStorage)
                    // && containsKnownFile(externalStorage)
                        ) {
                    Log.d(TAG, "Using externalStorage dir " + externalStorage);
                    return externalStorage;
                }
            } else if (Build.VERSION.SDK_INT >= 11 && Environment.isExternalStorageEmulated()) {
                Log.d(TAG, "Removing emulated external storage dir " + externalStorage);
                candidatePaths.remove(externalStorage);
            }
        }

        // If any directory contains our special test file, consider that the microSD card.
        if (KNOWNFILE != null) {
            for (File dir : candidatePaths) {
                Log.d(TAG, String.format(Locale.ROOT, "findSdCardPath: Looking for known file in candidate path, %s", dir));
                if (containsKnownFile(dir)) return dir;
            }
        }

        // If we don't find the known file, still try taking the first candidate.
        if (!candidatePaths.isEmpty()) {
            Log.d(TAG, "No definitive path to SD card; taking the first realistic candidate.");
            return candidatePaths.iterator().next();
        }

        // If no reasonable path was found, give up.
        return null;
    }

    /** Add each path to the collection. */
    private static void addStrings(LinkedHashSet<File> candidatePaths, String[] newPaths) {
        for (String path : newPaths) {
            addPath(path, null, candidatePaths);
        }
    }

    /** Add ancestor of each File to the collection. */
    private static void addAncestors(LinkedHashSet<File> candidatePaths, File[] files) {
        for (int i = files.length - 1; i >= 0; i--) {
            addPath(null, ancestor(files[i]), candidatePaths);
        }
    }

    /**
     * Add a new candidate directory path to our list, if it's not obviously wrong.
     * Supply path as either String or File object.
     * @param strNew path of directory to add (or null)
     * @param fileNew directory to add (or null)
     */
    private static void addPath(String strNew, File fileNew, Collection<File> paths) {
        // If one of the arguments is null, fill it in from the other.
        if (strNew == null) {
            if (fileNew == null) return;
            strNew = fileNew.getPath();
        } else if (fileNew == null) {
            fileNew = new File(strNew);
        }

        if (!paths.contains(fileNew) &&
                // Check for paths known not to be removable SD card.
                // The antipattern check can be redundant, depending on where this is called from.
                !pathAntiPattern.matcher(strNew).matches()) {

            // Eliminate candidate if not a directory or not fully accessible.
            if (fileNew.exists() && fileNew.isDirectory() && fileNew.canExecute()) {
                Log.d(TAG, "  Adding candidate path " + strNew);
                paths.add(fileNew);
            } else {
                Log.d(TAG, String.format(Locale.ROOT, "  Invalid path %s: exists: %b isDir: %b canExec: %b canRead: %b",
                        strNew, fileNew.exists(), fileNew.isDirectory(), fileNew.canExecute(), fileNew.canRead()));
            }
        }
    }

    private static final String ANDROID_DIR = File.separator + "Android";

    private static File ancestor(File dir) {
        // getExternalFilesDir() and getExternalStorageDirectory() typically something app-specific like
        //   /storage/sdcard1/Android/data/com.mybackuparchives.android/files
        // so we want the great-great-grandparent folder.
        if (dir == null) {
            return null;
        } else {
            String path = dir.getAbsolutePath();
            int i = path.indexOf(ANDROID_DIR);
            if (i == -1) {
                return dir;
            } else {
                return new File(path.substring(0, i));
            }
        }
    }

    /** Returns true iff dir contains the special test file.
     * Assumes that dir exists and is a directory. (Is this a necessary assumption?) */
    private static boolean containsKnownFile(File dir) {
        if (KNOWNFILE == null) return false;

        File knownFile = new File(dir, KNOWNFILE);
        return knownFile.exists();
    }

    private static Pattern
            /** Pattern that SD card device should match */
            devicePattern = Pattern.compile("/dev/(block/.*vold.*|fuse)|/mnt/.*"),
    /** Pattern that SD card mount path should match */
    pathPattern = Pattern.compile("/(mnt|storage|external_sd|extsd|_ExternalSD|Removable|.*MicroSD).*",
            Pattern.CASE_INSENSITIVE),
    /** Pattern that the mount path should not match.
     * 'emulated' indicates an internal storage location, so skip it.
     * 'asec' is an encrypted package file, decrypted and mounted as a directory. */
    pathAntiPattern = Pattern.compile(".*(/secure|/asec|/emulated).*"),
    /** These are expected fs types, including vfat. tmpfs is not OK.
     * fuse can be removable SD card (as on Moto E or Asus ZenPad), or can be internal (Huawei G610). */
    fsTypePattern = Pattern.compile(".*(fat|msdos|ntfs|ext[34]|fuse|sdcard|esdfs).*");
}

PS

  • <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />マニフェストを忘れないでください。APIレベル23以上では、必ずcheckSelfPermission/ を使用してくださいrequestPermissions
  • SDカード上にあると予想されるファイルまたはフォルダーがある場合は、KNOWNFILE = "myappfile"を設定します。これにより、検出がより正確になります。
  • 明らかに、あなたはの値をキャッシュしたいと思うでしょう findSdCardPath(),必要になるたびに再計算するのでなくする必要があります。
  • Log.d()上記のコードにはたくさんのロギング()があります。正しいパスが見つからない場合の診断に役立ちます。ログを記録したくない場合は、コメントアウトしてください。

反対投票者、この回答を改善する方法を提案できますか?
LarsH

1

私が見つけた唯一の実用的な解決策は、反射を使用するこれです

 /**
 * Get external sd card path using reflection
 * @param mContext
 * @param is_removable is external storage removable
 * @return
 */
private static String getExternalStoragePath(Context mContext, boolean is_removable) {

    StorageManager mStorageManager = (StorageManager) mContext.getSystemService(Context.STORAGE_SERVICE);
    Class<?> storageVolumeClazz = null;
    try {
        storageVolumeClazz = Class.forName("android.os.storage.StorageVolume");
        Method getVolumeList = mStorageManager.getClass().getMethod("getVolumeList");
        Method getPath = storageVolumeClazz.getMethod("getPath");
        Method isRemovable = storageVolumeClazz.getMethod("isRemovable");
        Object result = getVolumeList.invoke(mStorageManager);
        final int length = Array.getLength(result);
        for (int i = 0; i < length; i++) {
            Object storageVolumeElement = Array.get(result, i);
            String path = (String) getPath.invoke(storageVolumeElement);
            boolean removable = (Boolean) isRemovable.invoke(storageVolumeElement);
            if (is_removable == removable) {
                return path;
            }
        }
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }
    return null;
}

私は個人的にリフレクションの使用を好みません。なぜなら、Googleは新しいバージョンのAndroidにおける下位互換性を高く評価していないからです。
Behrouz.M

0

理由はわかりませんが、使用する前に、パブリックストレージディレクトリに作成されたファイルに対して.createNewFile()を呼び出す必要があります。フレームワークでは、そのメソッドのコメントは、それが役に立たないと言っています。ここにサンプルがあります...


 String myPath = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PODCASTS) + File.separator + "My Directory";
            final File myDir = new File(myPath);
            try {
                myDir.mkdirs();
            } catch (Exception ex) {
                Toast.makeText(this, "error: " + ex.getMessage(), Toast.LENGTH_LONG).show();
            }

        String fname = "whatever";
        File newFile = new File(myDir, fname);

        Log.i(TAG, "File exists --> " + newFile.exists()) //will be false  
    try {
            if (newFile.createNewFile()) {

                 //continue 

              } else {

                Log.e(TAG, "error creating file");

            }

        } catch (Exception e) {
            Log.e(TAG, e.toString());
        }


0

utilsメソッドを作成して、SDカードがデバイスで使用できるかどうかを確認し、使用可能な場合はデバイスでSDカードのパスを取得します。

以下の2つのメソッドを、必要なプロジェクトのクラスにコピーできます。それで全部です。

public String isRemovableSDCardAvailable() {
    final String FLAG = "mnt";
    final String SECONDARY_STORAGE = System.getenv("SECONDARY_STORAGE");
    final String EXTERNAL_STORAGE_DOCOMO = System.getenv("EXTERNAL_STORAGE_DOCOMO");
    final String EXTERNAL_SDCARD_STORAGE = System.getenv("EXTERNAL_SDCARD_STORAGE");
    final String EXTERNAL_SD_STORAGE = System.getenv("EXTERNAL_SD_STORAGE");
    final String EXTERNAL_STORAGE = System.getenv("EXTERNAL_STORAGE");

    Map<Integer, String> listEnvironmentVariableStoreSDCardRootDirectory = new HashMap<Integer, String>();
    listEnvironmentVariableStoreSDCardRootDirectory.put(0, SECONDARY_STORAGE);
    listEnvironmentVariableStoreSDCardRootDirectory.put(1, EXTERNAL_STORAGE_DOCOMO);
    listEnvironmentVariableStoreSDCardRootDirectory.put(2, EXTERNAL_SDCARD_STORAGE);
    listEnvironmentVariableStoreSDCardRootDirectory.put(3, EXTERNAL_SD_STORAGE);
    listEnvironmentVariableStoreSDCardRootDirectory.put(4, EXTERNAL_STORAGE);

    File externalStorageList[] = null;
    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT) {
        externalStorageList = getContext().getExternalFilesDirs(null);
    }
    String directory = null;
    int size = listEnvironmentVariableStoreSDCardRootDirectory.size();
    for (int i = 0; i < size; i++) {
        if (externalStorageList != null && externalStorageList.length > 1 && externalStorageList[1] != null)
            directory = externalStorageList[1].getAbsolutePath();
        else
            directory = listEnvironmentVariableStoreSDCardRootDirectory.get(i);

        directory = canCreateFile(directory);
        if (directory != null && directory.length() != 0) {
            if (i == size - 1) {
                if (directory.contains(FLAG)) {
                    Log.e(getClass().getSimpleName(), "SD Card's directory: " + directory);
                    return directory;
                } else {
                    return null;
                }
            }
            Log.e(getClass().getSimpleName(), "SD Card's directory: " + directory);
            return directory;
        }
    }
    return null;
}

/**
 * Check if can create file on given directory. Use this enclose with method
 * {@link BeginScreenFragement#isRemovableSDCardAvailable()} to check sd
 * card is available on device or not.
 * 
 * @param directory
 * @return
 */
public String canCreateFile(String directory) {
    final String FILE_DIR = directory + File.separator + "hoang.txt";
    File tempFlie = null;
    try {
        tempFlie = new File(FILE_DIR);
        FileOutputStream fos = new FileOutputStream(tempFlie);
        fos.write(new byte[1024]);
        fos.flush();
        fos.close();
        Log.e(getClass().getSimpleName(), "Can write file on this directory: " + FILE_DIR);
    } catch (Exception e) {
        Log.e(getClass().getSimpleName(), "Write file error: " + e.getMessage());
        return null;
    } finally {
        if (tempFlie != null && tempFlie.exists() && tempFlie.isFile()) {
            // tempFlie.delete();
            tempFlie = null;
        }
    }
    return directory;
}

-1

すべての外部デバイスで機能しますが、外部デバイスのフォルダー名のみを取得し、Fileクラスを使用して特定の場所からファイルを取得する必要があることを確認してください。

public static List<String> getExternalMounts() {
        final List<String> out = new ArrayList<>();
        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);
                    }
                }
            }
        }
        return out;
    }

呼び出し:

List<String> list=getExternalMounts();
        if(list.size()>0)
        {
            String[] arr=list.get(0).split("/");
            int size=0;
            if(arr!=null && arr.length>0) {
                size= arr.length - 1;
            }
            File parentDir=new File("/storage/"+arr[size]);
            if(parentDir.listFiles()!=null){
                File parent[] = parentDir.listFiles();

                for (int i = 0; i < parent.length; i++) {

                    // get file path as parent[i].getAbsolutePath());

                }
            }
        }

外部ストレージにアクセスする

外部ストレージのファイルを読み書きするには、アプリがREAD_EXTERNAL_STORAGEまたはWRITE_EXTERNAL_STORAGEシステム権限を取得する必要があります。例えば:

<manifest ...>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    ...
</manifest>

-2

/ sdcard =>内部ストレージ(これはシンボリックリンクですが動作するはずです)

/ mnt / extSdCard =>外部SDカード

これはサムスンギャラクシーS3用です

あなたはおそらくこれが真実であることを信じることができます...しかし、ダブルチェック!


8
私はいくつかの異なるAndroidフォンを使用しており、そのうちの約半分がSamsungであり、この場所が使用されるのを見たことがありません。それはS3にも当てはまるかもしれませんが、「おそらく、ほとんどの場合、これに当てはまることを期待できます」というのは完全に誤りです。
Geobits 2013

違う。/sdcard私のソニー2305の外部へのシンボリックリンクがある
jiggunjerの

2
私はそうではないかもしれないと言っていませんか?
robbyoconnor
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.