[Android] スマホのSDカードパスを取得、機種依存対策

SDカードはAndroidではリムーバブル ストレージとして外部ストレージの一部として規定されています。ストレージ オプション | Android Developers
 
android sd card

 



SDカードのパスを取得

 
このSDカードへのアクセスについては、メジャーアップデート毎に仕様が変更されているのが実情で、Android 2.xからAndroid 8までサポートしたいと言う場合には悲惨なことになります。
 
Android 7.0ではこのような方法で可能なようです。
特定のディレクトリへのアクセス
 
KitkatからNougatまではこの方法でできるようです。
How to get SD_Card path in android6.0 programmatically
 
Android 8.0では

Starting in Android O, the Storage Access Framework allows custom documents providers to create seekable file descriptors for files residing in a remote data source…

Ref: Universal way to write to external SD card on Android
 
まだ情報が足りませんんが、結局あまり使って欲しくないのが本音なのではと思いますね。そもそもNexusとかSDカードスロットがない
Cloudを使うか端末内の外部ストレージ(取り外し不可の)を使うのがベストでしょうか。
 
 
以下昔のSDカードのやり方など

 
Sdカードへの読み書きのヒストリーは

  • 当初はExternal Storageとして書き込みは通常で可能
  • JELLY_BEAN:これ以下では
    • どこにマウントされているか調べればSDカードの書き込める、端末依存
    • External Storageは内部メモリー領域を指す
  • KitKat: 書き込み完全不可
  • Lollipop: Storage Access FrameWorkにより可能となる

Android 5.0 ではストレージ アクセス フレームワークが拡張されて、再度SDカードのデータ読み書きができるようになりました。

とても厄介なのは、バージョン毎に異なる手法を取らざる得なくなり、正直SDカードで云々というのは避けたいところです。最近ではSDカードスロットが無いものもある

以下古い記事、使うこともあるかもしれませんので

Kitkat以降は完全にSDカードの書き込みは使えない

 
External Storage Technical Information によれば
http://source.android.com/devices/tech/storage/index.html
 
分かりにくい表現が使われていますが、結局こうありますね
Apps must not be allowed to write to secondary external storage devices
このsecondary external storageがSDカードを指しているのでしょう
 
しかし、ダメだと言われるとかえってなんとかしたくなるのが人情
 
http://androplus.kagome-kagome.com/Entry/148/
/etc/permissions/platform.xml を編集すれば使えるとか
ただし、要root なので…
 
こちらは隠しメソッドの getVolumeList() を使うらしいですが
http://qiita.com/aMasatoYui/items/e13664455af45123a66e
 
結局、世の中 Cloud の流れで、SDカードに無理に保存しようというのが
時代遅れなのかとも思います
 

JELLY_BEANでのExternal Storage

 
Androidでは、外部ストレージのパスを取得するために Environment.getExternalStorageDirectory() が用意されている

しかし、ストレージが内部ストレージ(取外不可)と外部ストレージ(取外可能)に別れている場合、

内部ストレージを取得するExternalStorageDirectory といいながら、内部メモリにある、
ユーザーがあたかもSDカードのように自由にアクセスできる領域を指しているだけである
本当の外部にあるストレージ、SDカードのパスを取得する Android API は用意されていない

SDカードのパスを取得するAPIは無い

Google developer: Using the External Storage

In this case, the SD card is not part of the external storage and your app cannot access it (the extra storage is intended only for user-provided media that the system scans).

それでもやはりSDカードの画像データを取り出したい、音楽データがSDにありそれを再生したいというような要望はつきません。何とかできないかと調べてみたら、幾つかトライされているものが、国内外にありました。

Ref: http://inujirushi123.blog.fc2.com/?no=93
システムの設定ファイルから取得する方法です

/system/etc/vold.fstab

システム設定ファイルの中身
#######################
## Regular device mount
##
## Format: dev_mount
## label – Label for the volume
## mount_point – Where the volume will be mounted
## part – Partition # (1 based), or ‘auto’ for first usable partition.
## – List of sysfs paths to source devices
######################

# Mounts the first usable partition of the specified device
dev_mount sdcard /storage/ext_sd auto /devices/platform/msm_sdcc.4/mmc_host
dev_mount usb /storage/usb auto /devices/platform/msm_hsusb_host

「/storage/ext_sd」が、取得するべきSDカードのマウント先(path)
「dev_mount」(機種によっては「fuse_mount」)で始まる行の半角スペース区切りで3番目がpathを示す
というのはどうやら共通の仕様のようだそうです

以下は環境変数での検索
Ref: http://inujirushi123.blog.fc2.com/blog-entry-30.html

Android APIには取得する術がないので環境変数からパスを取得
環境変数を呼び出すには System#getenv(String name) を使用

とても厄介なのは、バージョン毎に異なる手法を取らざる得なくなり、正直SDカードで云々というのは避けたいところです。最近ではSDカードスロットが無いものもある