Android 本地音乐播放(读取系统媒体库 + MediaPlayer)

Android 本地音乐播放(读取系统媒体库 + MediaPlayer)

简介

本文介绍如何在 Android 中读取手机本地音乐并进行播放,包含:

  • ContentResolver.query() 参数说明
  • 常见查询条件写法
  • MediaPlayer 的正确初始化与释放
  • 权限与常见崩溃点修复

适用场景:播放设备中的本地音频文件(非在线流媒体)。


一、读取手机中的音乐

本地音频数据通常来自 MediaStore,通过 ContentResolver 查询:

java 复制代码
Cursor cursor = contentResolver.query(
        contentUri,
        projection,
        selection,
        selectionArgs,
        sortOrder
);

query() 常见参数说明:

  1. contentUri:查询目标表,例如外部音频库:
    MediaStore.Audio.Media.EXTERNAL_CONTENT_URI

  2. projection:要返回的列(字段)

  3. selection:筛选条件(where 子句,不带 where 关键字)

  4. selectionArgs:给 selection? 占位符赋值

  5. sortOrder:排序方式

1) 常用 URI(修复原文拼写错误)

原文中的 Media.EXTERNL_CONTENT_URL 是错误写法,正确为:

java 复制代码
MediaStore.Audio.Media.EXTERNAL_CONTENT_URI

2) 推荐查询字段

java 复制代码
String[] projection = new String[]{
        MediaStore.Audio.Media._ID,
        MediaStore.Audio.Media.DISPLAY_NAME,
        MediaStore.Audio.Media.ARTIST,
        MediaStore.Audio.Media.ALBUM,
        MediaStore.Audio.Media.DURATION,
        MediaStore.Audio.Media.SIZE
};

说明:MediaStore.Audio.Media.DATA_data)在高版本中已不推荐直接使用,建议优先使用 content:// 类型的 Uri 来播放。

3) 推荐筛选条件

java 复制代码
String selection = MediaStore.Audio.Media.IS_MUSIC + "!= 0";
String sortOrder = MediaStore.Audio.Media.DATE_ADDED + " DESC";

复杂筛选建议使用 selectionArgs,避免拼接字符串导致可读性差或出错。

4) 查询示例(含资源释放)

java 复制代码
List<Uri> musicUris = new ArrayList<>();
Uri contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;

try (Cursor cursor = getContentResolver().query(
        contentUri,
        projection,
        selection,
        null,
        sortOrder
)) {
    if (cursor != null) {
        int idIndex = cursor.getColumnIndexOrThrow(MediaStore.Audio.Media._ID);
        while (cursor.moveToNext()) {
            long id = cursor.getLong(idIndex);
            Uri songUri = ContentUris.withAppendedId(contentUri, id);
            musicUris.add(songUri);
        }
    }
}

二、使用 MediaPlayer 播放

下面给出一个更安全的初始化与播放流程:

java 复制代码
private RecyclerView rcList;
private MediaPlayer mediaPlayer;
private final List<Uri> musicUris = new ArrayList<>();

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    rcList = findViewById(R.id.rc_list);
    rcList.setLayoutManager(new LinearLayoutManager(this));

    loadMusic(); // 查询本地音乐并填充 musicUris

    if (!musicUris.isEmpty()) {
        mediaPlayer = MediaPlayer.create(this, musicUris.get(0));
    }
}

private void play() {
    if (mediaPlayer != null && !mediaPlayer.isPlaying()) {
        mediaPlayer.start();
    }
}

private void pause() {
    if (mediaPlayer != null && mediaPlayer.isPlaying()) {
        mediaPlayer.pause();
    }
}

@Override
protected void onDestroy() {
    super.onDestroy();
    if (mediaPlayer != null) {
        mediaPlayer.release();
        mediaPlayer = null;
    }
}

修复点(对应原文常见 bug)

  • MediaPlayer.create(this, Uri.parse(path)) 并不总是可靠,优先使用 content://Uri
  • 播放前需要判空,避免 musicList.get(0) 越界
  • onDestroy() 中调用 release(),避免内存泄漏和句柄占用

三、权限说明

  • Android 6.0+ 需运行时权限(旧版本常见是读取存储权限)
  • Android 13+ 推荐使用 READ_MEDIA_AUDIO

未授权时直接查询媒体库,通常会得到空结果或抛出异常。


四、总结

本地音乐播放的核心流程是:

  1. 通过 MediaStore 查询音频数据
  2. 组装可播放的 content:// Uri
  3. 使用 MediaPlayer 播放并正确释放资源

如果后续你需要,我可以再帮你把这篇文档继续升级成:

  • 可直接运行的完整示例(含权限申请)
  • 支持上一首/下一首、进度条、后台播放
  • 使用 ExoPlayer 的现代实现版本

github仓库代码

复制代码
https://github.com/lixiangoko/Musicimpleness
相关推荐
小bo波3 小时前
使用Thread子类创建线程 VS 使用Runnable接口创建线程的区别
java·多线程·thread·并发编程·runnable
SamDeepThinking4 小时前
高并发场景下,CompletableFuture与ForkJoinPool该如何取舍?
java·后端·面试
张不才6 小时前
CPU 100% 了怎么办?Java 性能排障的标准化操作
java·后端
shepherd1118 小时前
吞吐量提升 10 倍:高并发大批量数据处理任务的架构演进与性能调优
java·后端·架构
plainGeekDev11 小时前
单例模式 → object 声明
android·java·kotlin
用户2986985301412 小时前
Java 实现 Word 文档文本与图片提取的方法
java·后端
SimonKing12 小时前
铁子,IntelliJ IDEA 2026.1.3来了,升不升?
java·后端·程序员
咖啡八杯1 天前
GoF设计模式——策略模式
java·后端·spring·设计模式
用户128526116021 天前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java