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
相关推荐
Javatutouhouduan12 小时前
2026Java面试的正确打开方式!
java·高并发·java面试·java面试题·后端开发·java编程·java八股文
JAVA面经实录91712 小时前
Java初级最终完整版学习路线图
java·spring·eclipse·maven
Cat_Rocky13 小时前
k8s-持久化存储,粗浅学习
java·学习·kubernetes
知识领航员14 小时前
蘑兔AI音乐深度实测:功能拆解、实测表现与适用场景
java·c语言·c++·人工智能·python·算法·github
释怀°Believe14 小时前
Spring解析
java·后端·spring
ooseabiscuit14 小时前
Laravel4.x:现代PHP框架的奠基之作
java·开发语言·php
节奏昂15 小时前
【一份基础软件的下载地址和安装地址】
java
没什么本事15 小时前
关于C# panel 添加lable问题 -- 明确X和Y 位置错误
android·java·c#
dhashdoia15 小时前
GPT-5.5 代码开发实战:Codex与Browser Use深度集成与星链4SAPI优化方案
java·数据库·人工智能·gpt·架构
xuhaoyu_cpp_java16 小时前
SpringMVC学习(二)
java·经验分享·笔记·学习·spring