web-dev-qa-db-ja.com

Android 4.0+の外部SDカードパスを取得するにはどうすればよいですか?

Samsung Galaxy S3には、/mnt/extSdCardにマウントされる外部SDカードスロットがあります。

私の質問は、Environment.getExternalStorageDirectory()のようなものでこのパスを取得する方法ですか?これによりmnt/sdcardが返され、外部SDカード用のAPIが見つかりません。 (または一部のタブレットのリムーバブルUSBストレージ)

ありがとうございました!

87

私が見つけた解決策のバリエーションがあります ここ

public static HashSet<String> getExternalMounts() {
    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);
                }
            }
        }
    }
    return out;
}

元の方法がテストされ、

  • Huawei X3(ストック)
  • Galaxy S2(ストック)
  • Galaxy S3(ストック)

これらがテストされたときにどのAndroidバージョンがオンであったかはわかりません。

変更したバージョンをテストしました

  • Moto Xoom 4.1.2(ストック)
  • OTGケーブルを使用したGalaxy Nexus(Cyanogenmod 10)
  • HTC Incredible(cyanogenmod 7.2)は、内部と外部の両方を返しました。このデバイスは、getExternalStorage()が代わりにsdcardへのパスを返すため、内部がほとんど使用されないという点で、ちょっと変わっています。

sDカードをメインストレージとして使用する単一のストレージデバイス

  • HTC G1(シアノゲンモッド6.1)
  • HTC G1(ストック)
  • HTC Vision/G2(ストック)

Incredibleを除き、これらすべてのデバイスはリムーバブルストレージのみを返しました。おそらく追加のチェックを行う必要がありますが、これはこれまでに見つけたどのソリューションよりも少なくとも少し優れています。

55
Gnathonic

システム内のすべてのSD-CARDへのパスを取得するより信頼性の高い方法を見つけました。これはすべてのAndroidバージョンで機能し、すべてのストレージ(エミュレートを含む)へのパスを返します。

すべてのデバイスで正常に動作します。

追伸:Environmentクラスのソースコードに基づきます。

private static final Pattern DIR_SEPORATOR = Pattern.compile("/");

/**
 * Raturns all available SD-Cards in the system (include emulated)
 *
 * Warning: Hack! Based on Android source code of version 4.3 (API 18)
 * Because there is no standart way to get it.
 * TODO: Test on future Android versions 4.4+
 *
 * @return paths to all available SD-Cards in the system (include emulated)
 */
public static String[] getStorageDirectories()
{
    // Final set of paths
    final Set<String> rv = new HashSet<String>();
    // Primary physical SD-CARD (not emulated)
    final String rawExternalStorage = System.getenv("EXTERNAL_STORAGE");
    // All Secondary SD-CARDs (all exclude primary) separated by ":"
    final String rawSecondaryStoragesStr = System.getenv("SECONDARY_STORAGE");
    // Primary emulated SD-CARD
    final String rawEmulatedStorageTarget = System.getenv("EMULATED_STORAGE_TARGET");
    if(TextUtils.isEmpty(rawEmulatedStorageTarget))
    {
        // Device has physical external storage; use plain paths.
        if(TextUtils.isEmpty(rawExternalStorage))
        {
            // EXTERNAL_STORAGE undefined; falling back to default.
            rv.add("/storage/sdcard0");
        }
        else
        {
            rv.add(rawExternalStorage);
        }
    }
    else
    {
        // Device has emulated storage; external storage paths should have
        // userId burned into them.
        final String rawUserId;
        if(Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR1)
        {
            rawUserId = "";
        }
        else
        {
            final String path = Environment.getExternalStorageDirectory().getAbsolutePath();
            final String[] folders = DIR_SEPORATOR.split(path);
            final String lastFolder = folders[folders.length - 1];
            boolean isDigit = false;
            try
            {
                Integer.valueOf(lastFolder);
                isDigit = true;
            }
            catch(NumberFormatException ignored)
            {
            }
            rawUserId = isDigit ? lastFolder : "";
        }
        // /storage/emulated/0[1,2,...]
        if(TextUtils.isEmpty(rawUserId))
        {
            rv.add(rawEmulatedStorageTarget);
        }
        else
        {
            rv.add(rawEmulatedStorageTarget + File.separator + rawUserId);
        }
    }
    // Add all secondary storages
    if(!TextUtils.isEmpty(rawSecondaryStoragesStr))
    {
        // All Secondary SD-CARDs splited into array
        final String[] rawSecondaryStorages = rawSecondaryStoragesStr.split(File.pathSeparator);
        Collections.addAll(rv, rawSecondaryStorages);
    }
    return rv.toArray(new String[rv.size()]);
}
53
Dmitriy Lozenko

私はあなたがこれを使用する必要がある外部SDカードを使用すると思います:

new File("/mnt/external_sd/")

OR

new File("/mnt/extSdCard/")

あなたの場合...

Environment.getExternalStorageDirectory()の代わりに

私のために働く。最初にディレクトリmntの内容を確認し、そこから作業する必要があります。


何らかの種類の選択方法を使用して、使用するSDカードを選択する必要があります。

File storageDir = new File("/mnt/");
if(storageDir.isDirectory()){
    String[] dirList = storageDir.list();
    //TODO some type of selecton method?
}
32
FabianCook

すべての 外部ストレージ (それらが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ドライブを除外するために、セカンダリディレクトリのリストをフィルタリングすることをお勧めします。

いずれにせよ、ハードコーディングされたパスを使用することは常に悪いアプローチであることに注意してください(特にすべてのメーカーが喜んで変更する場合)。

15
Paolo Rovelli

Dmitriy Lozenkoのソリューションを使用して、Asus Zenfone2Marshmallow 6.0.1およびソリューションが機能していません。ソリューションは、EMULATED_STORAGE_TARGETを取得するときに失敗しました。具体的には、microSDパスの場合:/ storage/F99C-10F4 /。コードを編集して、エミュレートされたアプリケーションパスからエミュレートされたルートパスをcontext.getExternalFilesDirs(null);で直接取得し、既知のphone-model-specific物理パスを追加しました。

生活を楽にするために、ライブラリを作成しました here 。 gradle、maven、sbt、leiningenビルドシステム経由で使用できます。

昔ながらの方法が好きな場合は、 here から直接ファイルをコピーして貼り付けることもできますが、手動でチェックせずに将来更新があるかどうかはわかりません。

ご質問やご提案がありましたら、お知らせください

14
HendraWD

良いニュースです! KitKatには、これらのセカンダリ共有ストレージデバイスとやり取りするためのパブリックAPIがあります。

新しいContext.getExternalFilesDirs()およびContext.getExternalCacheDirs()メソッドは、プライマリデバイスとセカンダリデバイスの両方を含む複数のパスを返すことができます。その後、それらを反復処理し、Environment.getStorageState()およびFile.getFreeSpace()を確認して、ファイルを保存する最適な場所を決定します。これらのメソッドは、support-v4ライブラリのContextCompatでも使用できます。

また、Contextによって返されるディレクトリの使用のみに関心がある場合は、READ_またはWRITE_EXTERNAL_STORAGE権限は不要になります。今後は、これらのディレクトリへの読み取り/書き込みアクセスが常に許可され、追加のアクセス許可は必要ありません。

アプリは、次のように許可リクエストを終了することで、古いデバイスでも引き続き動作できます。

<uses-permission
    Android:name="Android.permission.WRITE_EXTERNAL_STORAGE"
    Android:maxSdkVersion="18" />
12
Jeff Sharkey

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)
        return file.getAbsolutePath();
      file=parentFile;
      }
    }
9

すべての外部SDカードにアクセスするために、次のことを行いました。

と:

File primaryExtSd=Environment.getExternalStorageDirectory();

プライマリ外部SDへのパスを取得してから、次を使用します。

File parentDir=new File(primaryExtSd.getParent());

プライマリ外部ストレージの親ディレクトリを取得し、すべての外部sdの親でもあります。これで、すべてのストレージをリストし、必要なストレージを選択できます。

役に立てば幸いです。

9
valenta

皆さん、特に@SmartLemonから提供された手がかりに感謝します。他の誰かがそれを必要とする場合に備えて、私は最終的な解決策をここに入れます(最初にリストされた外部SDカードを見つけるため)

public File getExternalSDCardDirectory()
{
    File innerDir = Environment.getExternalStorageDirectory();
    File rootDir = innerDir.getParentFile();
    File firstExtSdCard = innerDir ;
    File[] files = rootDir.listFiles();
    for (File file : files) {
        if (file.compareTo(innerDir) != 0) {
            firstExtSdCard = file;
            break;
        }
    }
    //Log.i("2", firstExtSdCard.getAbsolutePath().toString());
    return firstExtSdCard;
}

外部SDカードがない場合、オンボードストレージを返します。 SDカードが存在しない場合は使用します。変更する必要がある場合があります。

5
rml

私のコードを参照してください、あなたに役立つことを願っています:

    Runtime runtime = Runtime.getRuntime();
    Process proc = runtime.exec("mount");
    InputStream is = proc.getInputStream();
    InputStreamReader isr = new InputStreamReader(is);
    String line;
    String mount = new String();
    BufferedReader br = new BufferedReader(isr);
    while ((line = br.readLine()) != null) {
        if (line.contains("secure")) continue;
        if (line.contains("asec")) continue;

        if (line.contains("fat")) {//TF card
            String columns[] = line.split(" ");
            if (columns != null && columns.length > 1) {
                mount = mount.concat("*" + columns[1] + "\n");
            }
        } else if (line.contains("Fuse")) {//internal storage
            String columns[] = line.split(" ");
            if (columns != null && columns.length > 1) {
                mount = mount.concat(columns[1] + "\n");
            }
        }
    }
    txtView.setText(mount);
3
cst05001

実際、一部のデバイスでは、外部SDカードのデフォルト名がextSdCardとして表示され、その他のデバイスではsdcard1です。

このコードスニペットは、その正確なパスを見つけるのに役立ち、外部デバイスのパスを取得するのに役立ちます。

String sdpath,sd1path,usbdiskpath,sd0path;    
        if(new File("/storage/extSdCard/").exists())
            {
               sdpath="/storage/extSdCard/";
               Log.i("Sd Cardext Path",sdpath);
            }
        if(new File("/storage/sdcard1/").exists())
         {
              sd1path="/storage/sdcard1/";
              Log.i("Sd Card1 Path",sd1path);
         }
        if(new File("/storage/usbcard1/").exists())
         {
              usbdiskpath="/storage/usbcard1/";
              Log.i("USB Path",usbdiskpath);
         }
        if(new File("/storage/sdcard0/").exists())
         {
              sd0path="/storage/sdcard0/";
              Log.i("Sd Card0 Path",sd0path);
         }
2
Sam

はい。メーカーによっては、Samsung Tab 3のextsdのように異なるSDカード名が使用され、他のサムスンデバイスは、この異なるメーカーが異なる名前を使用するようにsdcardを使用します。

あなたと同じ要件がありました。だから私はあなたのプロジェクトからあなたのためのサンプル例を作成しましたこのリンクに行きます Android Directory chooser androi-dirchooserライブラリを使用する例。この例では、SDカードを検出し、すべてのサブフォルダーを一覧表示します。また、デバイスに複数のSDカードがあるかどうかも検出します。

コードの一部は次のようになります。完全な例については、リンクに移動してください Android Directory Chooser

/**
* Returns the path to internal storage ex:- /storage/emulated/0
 *
* @return
 */
private String getInternalDirectoryPath() {
return Environment.getExternalStorageDirectory().getAbsolutePath();
 }

/**
 * Returns the SDcard storage path for samsung ex:- /storage/extSdCard
 *
 * @return
 */
    private String getSDcardDirectoryPath() {
    return System.getenv("SECONDARY_STORAGE");
}


 mSdcardLayout.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View view) {
        String sdCardPath;
        /***
         * Null check because user may click on already selected buton before selecting the folder
         * And mSelectedDir may contain some wrong path like when user confirm dialog and swith back again
         */

        if (mSelectedDir != null && !mSelectedDir.getAbsolutePath().contains(System.getenv("SECONDARY_STORAGE"))) {
            mCurrentInternalPath = mSelectedDir.getAbsolutePath();
        } else {
            mCurrentInternalPath = getInternalDirectoryPath();
        }
        if (mCurrentSDcardPath != null) {
            sdCardPath = mCurrentSDcardPath;
        } else {
            sdCardPath = getSDcardDirectoryPath();
        }
        //When there is only one SDcard
        if (sdCardPath != null) {
            if (!sdCardPath.contains(":")) {
                updateButtonColor(STORAGE_EXTERNAL);
                File dir = new File(sdCardPath);
                changeDirectory(dir);
            } else if (sdCardPath.contains(":")) {
                //Multiple Sdcards show root folder and remove the Internal storage from that.
                updateButtonColor(STORAGE_EXTERNAL);
                File dir = new File("/storage");
                changeDirectory(dir);
            }
        } else {
            //In some unknown scenario at least we can list the root folder
            updateButtonColor(STORAGE_EXTERNAL);
            File dir = new File("/storage");
            changeDirectory(dir);
        }


    }
});
2
Shivaraj Patil

このソリューション(この質問に対する他の回答から集められた)は、System.getenv("SECONDARY_STORAGE")がMarshmallowで役に立たないという事実(@onoが述べたように)を処理します。

テスト済みおよび作業中:

  • Samsung Galaxy Tab 2(Android 4.1.1-ストック)
  • Samsung Galaxy Note 8.0(Android 4.2.2-ストック)
  • Samsung Galaxy S4(Android 4.4-ストック)
  • Samsung Galaxy S4(Android 5.1.1-Cyanogenmod)
  • Samsung Galaxy Tab A(Android 6.0.1-ストック)

    /**
     * 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
DaveAlden

一部のデバイス(samsung galaxy sIIなど)では、内部メモリカードはvfatになっています。この場合、最後のコードを参照して、パス内部メモリカード(/ mnt/sdcad)を取得しますが、外部カードは取得しません。以下のコードでこの問題を解決します。

static String getExternalStorage(){
         String exts =  Environment.getExternalStorageDirectory().getPath();
         try {
            FileReader fr = new FileReader(new File("/proc/mounts"));       
            BufferedReader br = new BufferedReader(fr);
            String sdCard=null;
            String line;
            while((line = br.readLine())!=null){
                if(line.contains("secure") || line.contains("asec")) continue;
            if(line.contains("fat")){
                String[] pars = line.split("\\s");
                if(pars.length<2) continue;
                if(pars[1].equals(exts)) continue;
                sdCard =pars[1]; 
                break;
            }
        }
        fr.close();
        br.close();
        return sdCard;  

     } catch (Exception e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    return null;
}
1
Alex
       File[] files = null;
    File file = new File("/storage");// /storage/emulated
if (file.exists()) {
        files = file.listFiles();
            }
            if (null != files)
                for (int j = 0; j < files.length; j++) {
                    Log.e(TAG, "" + files[j]);
                    Log.e(TAG, "//--//--// " +             files[j].exists());

                    if (files[j].toString().replaceAll("_", "")
                            .toLowerCase().contains("extsdcard")) {
                        external_path = files[j].toString();
                        break;
                    } else if (files[j].toString().replaceAll("_", "")
                            .toLowerCase()
                            .contains("sdcard".concat(Integer.toString(j)))) {
                        // external_path = files[j].toString();
                    }
                    Log.e(TAG, "--///--///--  " + external_path);
                }
1
Amit

SDカードのファイルにアクセスするには、HTC One X(Android)で次のパスを使用します。

file:///storage/sdcard0/folder/filename.jpg

トリプル "/"!

0
HTC user

Context.getExternalCacheDirs()またはContext.getExternalFilesDirs()またはContext.getObbDirs()のようなものを使用できます。これらは、アプリケーションがファイルを保存できるすべての外部ストレージデバイスにアプリケーション固有のディレクトリを提供します。

そのため、このようなもの-Context.getExternalCacheDirs()[i] .getParentFile()。getParentFile()。getParentFile()。getParent()は、外部ストレージデバイスのルートパスを取得できます。

これらのコマンドは別の目的のためのものですが、他の答えはうまくいきませんでした。

このリンクは良いポインタをくれました- https://possiblemobile.com/2014/03/Android-external-storage/

0
Sharath Holla

私はこのコードがあなたの問題を確実に解決すると確信しています...これは私のためにうまく機能しています... \

try {
            File mountFile = new File("/proc/mounts");
            usbFoundCount=0;
            sdcardFoundCount=0;
            if(mountFile.exists())
             {
                Scanner usbscanner = new Scanner(mountFile);
                while (usbscanner.hasNext()) {
                    String line = usbscanner.nextLine();
                    if (line.startsWith("/dev/Fuse /storage/usbcard1")) {
                        usbFoundCount=1;
                        Log.i("-----USB--------","USB Connected and properly mounted---/dev/Fuse /storage/usbcard1" );
                    }
            }
         }
            if(mountFile.exists()){
                Scanner sdcardscanner = new Scanner(mountFile);
                while (sdcardscanner.hasNext()) {
                    String line = sdcardscanner.nextLine();
                    if (line.startsWith("/dev/Fuse /storage/sdcard1")) {
                        sdcardFoundCount=1;
                        Log.i("-----USB--------","USB Connected and properly mounted---/dev/Fuse /storage/sdcard1" );
                    }
            }
         }
            if(usbFoundCount==1)
            {
                Toast.makeText(context,"USB Connected and properly mounted", 7000).show();
                Log.i("-----USB--------","USB Connected and properly mounted" );
            }
            else
            {
                Toast.makeText(context,"USB not found!!!!", 7000).show();
                Log.i("-----USB--------","USB not found!!!!" );

            }
            if(sdcardFoundCount==1)
            {
                Toast.makeText(context,"SDCard Connected and properly mounted", 7000).show();
                Log.i("-----SDCard--------","SDCard Connected and properly mounted" );
            }
            else
            {
                Toast.makeText(context,"SDCard not found!!!!", 7000).show();
                Log.i("-----SDCard--------","SDCard not found!!!!" );

            }
        }catch (Exception e) {
            e.printStackTrace();
        } 
0
Sam

それは真実ではない。/mnt/sdcard/external_sdは、SDカードがマウントされていない場合でも存在できます。マウントされていないときに/ mnt/sdcard/external_sdに書き込もうとすると、アプリケーションがクラッシュします。

sDカードが最初にマウントされているかどうかを確認する必要があります:

boolean isSDPresent = Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED);
0
//manifest file outside the application tag
//please give permission write this 
//<uses-permission Android:name="Android.permission.READ_EXTERNAL_STORAGE"/>
        File file = new File("/mnt");
        String[] fileNameList = file.list(); //file names list inside the mnr folder
        String all_names = ""; //for the log information
        String foundedFullNameOfExtCard = ""; // full name of ext card will come here
        boolean isExtCardFounded = false;
        for (String name : fileNameList) {
            if (!isExtCardFounded) {
                isExtCardFounded = name.contains("ext");
                foundedFullNameOfExtCard = name;
            }
            all_names += name + "\n"; // for log
        }
        Log.d("dialog", all_names + foundedFullNameOfExtCard);
0
String secStore = System.getenv("SECONDARY_STORAGE");

File externalsdpath = new File(secStore);

これにより、外部SDセカンダリストレージのパスが取得されます。

0
 String path = Environment.getExternalStorageDirectory()
                        + File.separator + Environment.DIRECTORY_PICTURES;
                File dir = new File(path);
0
sarra srairi

System.getenv("SECONDARY_STORAGE")は、Marshmallowに対してnullを返します。これは、すべての外部要素を見つける別の方法です。あなたはそれが内部/外部かどうかを決定する取り外し可能かどうかを確認することができます

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
    File[] externalCacheDirs = context.getExternalCacheDirs();
    for (File file : externalCacheDirs) {
        if (Environment.isExternalStorageRemovable(file)) {
            // It's a removable storage
        }
    }
}
0
ono

Dmitriy LozenkoおよびGnathonicで提供されるソリューションを試しましたSamsung Galaxy Tab S2(モデル:T819Y)外部SDカードディレクトリ。 mountコマンドの実行には、外部SDカードディレクトリへの必要なパスが含まれていました(/ Storage/A5F9-15F4)ですが、正規表現と一致しなかったため、返されませんでした。 Samsungに続くディレクトリ命名メカニズムが得られません。なぜ標準から逸脱しているのか(i.e。extsdcard)そして私の場合のように本当に怪しいものを考え出す(i.e。/ Storage/A5F9-15F4)不足しているものはありますか?とにかく、正規表現のGnathonic'sソリューションの次の変更により、有効なSDカードディレクトリを取得できました。

final HashSet<String> out = new HashSet<String>();
        String reg = "(?i).*(vold|media_rw).*(sdcard|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;

これが有効なソリューションであるかどうか、他のSamsungタブレットでも結果が得られるかどうかはわかりませんが、今のところ問題は解決しています。以下は、Android(v6.0)でリムーバブルSDカードのパスを取得する別の方法です。 Android Marshmallowでメソッドをテストしましたが、動作します。使用されるアプローチは非常に基本的であり、他のバージョンでも確実に機能しますが、テストは必須です。それに関するいくつかの洞察が役立ちます:

public static String getSDCardDirPathForAndroidMarshmallow() {

    File rootDir = null;

    try {
        // Getting external storage directory file
        File innerDir = Environment.getExternalStorageDirectory();

        // Temporarily saving retrieved external storage directory as root
        // directory
        rootDir = innerDir;

        // Splitting path for external storage directory to get its root
        // directory

        String externalStorageDirPath = innerDir.getAbsolutePath();

        if (externalStorageDirPath != null
                && externalStorageDirPath.length() > 1
                && externalStorageDirPath.startsWith("/")) {

            externalStorageDirPath = externalStorageDirPath.substring(1,
                    externalStorageDirPath.length());
        }

        if (externalStorageDirPath != null
                && externalStorageDirPath.endsWith("/")) {

            externalStorageDirPath = externalStorageDirPath.substring(0,
                    externalStorageDirPath.length() - 1);
        }

        String[] pathElements = externalStorageDirPath.split("/");

        for (int i = 0; i < pathElements.length - 1; i++) {

            rootDir = rootDir.getParentFile();
        }

        File[] files = rootDir.listFiles();

        for (File file : files) {
            if (file.exists() && file.compareTo(innerDir) != 0) {

                // Try-catch is implemented to prevent from any IO exception
                try {

                    if (Environment.isExternalStorageRemovable(file)) {
                        return file.getAbsolutePath();

                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }

            }
        }
    } catch (Exception ex) {
        ex.printStackTrace();
    }
    return null;
}

この問題を処理する他のアプローチがある場合は、親切に共有してください。ありがとう

0
Abdul Rehman