轉載請注明出處: http://blog.csdn.net/zhaokaiqiang1992
今天一打開博客,發現一位朋友給我發了一封私信,請教下面的問題,所以特寫此文章,為其解惑。
從這位朋友的提問中,我們需要了解下面的這些問題,才能給他很好的解答:
(1)如何獲取手機里所有歌曲的信息?
(2)在歌曲文件發生改變,比如增刪操作之后,如何及時的更新媒體庫,從而獲取到最新的歌曲信息?
(3)在4.4版本之后,掃描sd卡,更新媒體庫的操作發生變化了嗎?
下面,我將就以上三個問題,進行解答。
(1)如何獲取手機里所有歌曲的信息?
如果要解決這個問題,那么我們首先要知道在Android系統中,是如何對歌曲信息進行管理的。
在Android中,系統為多媒體類型的文件(比如圖片、音頻、視頻等)建立了數據庫(sqlite數據庫),從而完成多媒體數據的維護工作。我們當然可以不用這些系統的數據庫,比如說,如果我們想獲取所有歌曲,我們可以掃描sd上所有的文件夾中的文件,然后根據文件的后綴名,就可以取到我們想要的mp3、wma文件等。但是,這樣的操作是非常效率低下的,所以是行不通的。
Android系統為我們建立起多媒體數據庫之后,便把多媒體常用的信息,比如歌曲名、文件大小、播放時長、專輯、歌手等常用信息保存在了數據庫里,那我們可以直接用多媒體庫中的數據,完成這個需求。雖然我們需要用多媒體庫,但是我們不能直接操作。Android為這些常用的需要共享的數據(多媒體和聯系人等),創建了ContentProvider,因此,如果我們想獲取到這些信息,我們就需要用ContentProvider。
在開始介紹之前,先給出需要用到的歌曲的實體類
/** * * @ClassName: com.example.mediastore.Song * @Description: 歌曲實體類 * @author zhaokaiqiang * @date 2014-12-4 上午11:49:59 * */ public class Song { private String fileName; private String title; private int duration; private String singer; private String album; private String year; private String type; private String size; private String fileUrl; public String getFileName() { return fileName; } public void setFileName(String fileName) { this.fileName = fileName; } public String getTitle() { return title; } public void setTitle(String title) { this.title = title; } public int getDuration() { return duration; } public void setDuration(int duration) { this.duration = duration; } public String getSinger() { return singer; } public void setSinger(String singer) { this.singer = singer; } public String getAlbum() { return album; } public void setAlbum(String album) { this.album = album; } public String getYear() { return year; } public void setYear(String year) { this.year = year; } public String getType() { return type; } public void setType(String type) { this.type = type; } public String getSize() { return size; } public void setSize(String size) { this.size = size; } public String getFileUrl() { return fileUrl; } public void setFileUrl(String fileUrl) { this.fileUrl = fileUrl; } public Song() { super(); } public Song(String fileName, String title, int duration, String singer, String album, String year, String type, String size, String fileUrl) { super(); this.fileName = fileName; this.title = title; this.duration = duration; this.singer = singer; this.album = album; this.year = year; this.type = type; this.size = size; this.fileUrl = fileUrl; } @Override public String toString() { return "Song [fileName=" + fileName + ", title=" + title + ", duration=" + duration + ", singer=" + singer + ", album=" + album + ", year=" + year + ", type=" + type + ", size=" + size + ", fileUrl=" + fileUrl + "]"; } }
有了上面的這些信息,我們完全可以做一個播放器了!
有了實體類之后,我封裝了一個類,專門用來獲取歌曲信息,下面是實現的代碼
/** * * @ClassName: com.example.mediastore.AudioUtils * @Description: 音頻文件幫助類 * @author zhaokaiqiang * @date 2014-12-4 上午11:39:45 * */ public class AudioUtils { /** * 獲取sd卡所有的音樂文件 * * @return * @throws Exception */ public static ArrayList<Song> getAllSongs(Context context) { ArrayList<Song> songs = null; Cursor cursor = context.getContentResolver().query( MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, new String[] { MediaStore.Audio.Media._ID, MediaStore.Audio.Media.DISPLAY_NAME, MediaStore.Audio.Media.TITLE, MediaStore.Audio.Media.DURATION, MediaStore.Audio.Media.ARTIST, MediaStore.Audio.Media.ALBUM, MediaStore.Audio.Media.YEAR, MediaStore.Audio.Media.MIME_TYPE, MediaStore.Audio.Media.SIZE, MediaStore.Audio.Media.DATA }, MediaStore.Audio.Media.MIME_TYPE + "=? or " + MediaStore.Audio.Media.MIME_TYPE + "=?", new String[] { "audio/mpeg", "audio/x-ms-wma" }, null); songs = new ArrayList<Song>(); if (cursor.moveToFirst()) { Song song = null; do { song = new Song(); // 文件名 song.setFileName(cursor.getString(1)); // 歌曲名 song.setTitle(cursor.getString(2)); // 時長 song.setDuration(cursor.getInt(3)); // 歌手名 song.setSinger(cursor.getString(4)); // 專輯名 song.setAlbum(cursor.getString(5)); // 年代 if (cursor.getString(6) != null) { song.setYear(cursor.getString(6)); } else { song.setYear("未知"); } // 歌曲格式 if ("audio/mpeg".equals(cursor.getString(7).trim())) { song.setType("mp3"); } else if ("audio/x-ms-wma".equals(cursor.getString(7).trim())) { song.setType("wma"); } // 文件大小 if (cursor.getString(8) != null) { float size = cursor.getInt(8) / 1024f / 1024f; song.setSize((size + "").substring(0, 4) + "M"); } else { song.setSize("未知"); } // 文件路徑 if (cursor.getString(9) != null) { song.setFileUrl(cursor.getString(9)); } songs.add(song); } while (cursor.moveToNext()); cursor.close(); } return songs; } }
代碼的思路很簡單,我們需要根據ContentResover獲取到一個Cursor,然后根據這個游標,遍歷所有的歌曲的信息。在上面的代碼中,我們查詢出了包括歌名、路徑、文件大小等在內的共10項數據,對於一般的應用這些足夠了。查詢出來之后,我們把信息轉換成了實體類,這樣操作起來更加方便。
如果要使用這個工具類,記得添加權限 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
(2)如何及時更新媒體庫
Android系統刷新媒體庫的數據的時機,是在開機的時候,即手機一開機,系統便重新掃描一下sd卡,並將多媒體數據庫更新一下。如果用戶刪除了某一個音頻文件,不重新開機的話,數據庫中的數據是不會更新的。那么,如果我們想用戶一打開軟件,就強制的更新多媒體數據庫,應該怎么做呢?
在4.4版本之前,我們可以使用發送廣播的方式,強制刷新多媒體庫
IntentFilter intentFilter = new IntentFilter( Intent.ACTION_MEDIA_SCANNER_STARTED); intentFilter.addAction(Intent.ACTION_MEDIA_SCANNER_FINISHED); intentFilter.addDataScheme("file"); sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://" + Environment.getExternalStorageDirectory() .getAbsolutePath())));
發送廣播之后,還需要注冊一個廣播接受者,來接受並處理掃描開始和結束事件
private class ScanReceiver extends BroadcastReceiver { public void onReceive(Context context, Intent intent) { String action = intent.getAction(); // 當系統開始掃描sd卡時,為了用戶體驗,可以加上一個等待框 if (Intent.ACTION_MEDIA_SCANNER_STARTED.equals(action)) { } // 當系統掃描完畢時,停止顯示等待框,並重新查詢ContentProvider if (Intent.ACTION_MEDIA_SCANNER_FINISHED.equals(action)) { } } }
通過這種方式,我們便可以強制更新媒體庫。
但是,在4.4之后,Android對一些操作的權限提高,如果在4.4的系統上使用這種方式,便會出現下面的錯誤
Caused by: java.lang.SecurityException: Permission Denial: not allowed to send broadcast android.intent.action.MEDIA_MOUNTED from pid=22360, uid=10163
這是因為在4.4之后,這個廣播只有系統應用才能發出,因此,我們不能使用這種方式了,我們可以使用下面的代碼實現相同的功能:
MediaScannerConnection.scanFile(this, new String[] { Environment .getExternalStorageDirectory().getAbsolutePath() }, null, null);
使用MediaScannerConnection的scanFile方法,就可以強制掃描我們需要更新的文件路徑,之后媒體數據庫也會同步更新,這樣,就不會出現文件刪除,在媒體庫中卻能搜索到的情況了,也能解決這位朋友提出的新增加歌曲的信息獲取問題了。