Android Media3 (一)— 简单播放视频

本文介绍了如何在Android应用中使用JetPackMedia3库,结合ExoPlayer进行视频和音频播放,包括添加依赖、配置PlayerView和实现视频播放的基本步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在App开发中偶尔会需要播放视频,JetPack Media3是目前官方主推的视频播放库,使用ExoPlayer作为默认播放器,让开发人员可以轻松的实现播放视频和音频功能。本文简单介绍下如何使用Media3库播放视频。

官方文档

添加依赖

在app module下的build.gradle中添加代码,如下:
 

dependencies {
    implementation("androidx.media3:media3-ui:1.1.0")
    implementation("androidx.media3:media3-session:1.1.0")
    implementation("androidx.media3:media3-exoplayer:1.1.0")
}

添加PlayerView

在布局文件中添加PlayerView,承载视频播放器,代码如下:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <androidx.media3.ui.PlayerView
        android:id="@+id/play_view"
        android:layout_width="match_parent"
        android:layout_height="320dp"
        app:layout_constraintTop_toTopOf="parent"
        app:resize_mode="fill"
        app:show_buffering="always"
        app:use_controller="false" />
        
</androidx.constraintlayout.widget.ConstraintLayout>

一些可配置参数如下:

插图参数:

控制器参数:

PlayerView参数:

播放视频

创建ExoPlayer配置到PlayerView中,设置好要播放的资源后就可以播放视频,代码如下:

class Media3ExampleActivity : AppCompatActivity() {

    private lateinit var binding: LayoutMedia3ExampleActivityBinding

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = LayoutMedia3ExampleActivityBinding.inflate(layoutInflater)
        setContentView(binding.root)
        binding.includeTitle.tvTitle.text = "Media3 Example"
        // 创建ExoPlayer,配置到PlayerView中
        binding.playView.player = ExoPlayer.Builder(this)
            .build()
        binding.playView.player?.run {
            // 设置播放监听
            addListener(object : Player.Listener {
                override fun onIsPlayingChanged(isPlaying: Boolean) {
                    super.onIsPlayingChanged(isPlaying)
                    // 播放状态变化回调
                }

                override fun onPlaybackStateChanged(playbackState: Int) {
                    super.onPlaybackStateChanged(playbackState)
                    when (playbackState) {
                        Player.STATE_IDLE -> {
                            //播放器停止时的状态
                        }

                        Player.STATE_BUFFERING -> {
                            // 正在缓冲数据
                        }

                        Player.STATE_READY -> {
                            // 可以开始播放
                        }

                        Player.STATE_ENDED -> {
                            // 播放结束
                        }
                    }

                }

                override fun onPlayerError(error: PlaybackException) {
                    super.onPlayerError(error)
                    // 获取播放错误信息
                }
            })
            // 设置重复模式
            // Player.REPEAT_MODE_ALL 无限重复
            // Player.REPEAT_MODE_ONE 重复一次
            // Player.REPEAT_MODE_OFF 不重复
            repeatMode = Player.REPEAT_MODE_ALL
            // 设置当缓冲完毕后直接播放视频
            playWhenReady = true
        }
        binding.btnPlaySingleVideo.setOnClickListener {
            binding.playView.player?.run {
                // 停止之前播放的视频
                stop()
                //设置单个资源
                setMediaItem(MediaItem.fromUri("https://minigame.vip/Uploads/images/2021/09/18/1631951892_page_img.mp4"))
                // 开始缓冲
                prepare()
            }
        }
        binding.btnPlayMultiVideo.setOnClickListener {
            binding.playView.player?.run {
                // 停止之前播放的视频
                stop()
                // 设置多个资源,当一个视频播完后自动播放下一个
                setMediaItems(arrayListOf(
                    MediaItem.fromUri("https://minigame.vip/Uploads/images/2021/09/18/1631951892_page_img.mp4"),
                    MediaItem.fromUri("https://storage.googleapis.com/exoplayer-test-media-1/mp4/dizzy-with-tx3g.mp4")
                ))
                // 开始缓冲
                prepare()
            }
        }
    }

    override fun onResume() {
        super.onResume()
        // 恢复播放
        binding.playView.onResume()
    }

    override fun onPause() {
        super.onPause()
        // 暂停播放
        binding.playView.onPause()
    }

    override fun onDestroy() {
        super.onDestroy()
        // 释放播放器资源
        binding.playView.player?.release()
        binding.playView.player = null
    }
}

效果如图:

示例

演示代码已在示例Demo中添加。

ExampleDemo github

ExampleDemo gitee

原文 Android Media3 (一)- 简单播放视频

★文末名片可以免费领取音视频开发学习资料,内容包括(FFmpeg ,webRTC ,rtmp ,hls ,rtsp ,ffplay ,srs)以及音视频学习路线图等等。

见下方!↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓

 

<think>我们正在处理个关于AndroidMedia3播放本地视频的请求。用户需要教程或示例代码。根据之前的引用,有个关于Android本地视频播放器开发的教程,但它是基于MediaPlayer类的,而用户现在要求使用Media3库。因此,我们需要提供使用AndroidMedia3(ExoPlayer)播放本地视频的指导。注意:Media3AndroidJetpack部分,它包含了ExoPlayer作为其播放器实现。我们将使用ExoPlayer来播放本地视频。步骤:1.添加依赖项2.创建播放器视图(PlayerView)布局3.在Activity中初始化ExoPlayer4.设置本地视频URI5.准备播放器并开始播放6.释放播放器我们将按照这些步骤提供示例代码。注意:用户需要本地视频,因此我们将使用本地文件的URI(例如:在assets目录或设备存储中的文件)。这里我们假设视频文件位于设备的内部存储中(例如:在应用的files目录下),或者我们将演示如何播放assets目录中的视频。由于播放assets目录中的视频需要特殊处理(通过AssetDataSource),我们将分别展示两种方式。但请注意,直接播放assets目录中的视频在ExoPlayer中并不直接支持,因为AssetDataSource需要被注入到数据源工厂中。不过,我们可以使用自定义DataSource.Factory来实现。然而,更常见的做法是将视频文件放在res/raw目录下,或者放在设备存储中。我们选择两种常见方式:方式1:播放设备存储中的视频(例如:sdcard上的视频)方式2:播放res/raw目录中的视频由于用户要求的是本地视频,我们优先展示从存储中读取的方式,因为res/raw目录通常用于较小的资源文件。步骤详解:1.添加依赖项:在build.gradle(Module级别)中添加Media3依赖。2.布局文件中添加PlayerView。3.在Activity中:创建ExoPlayer实例。将播放器绑定到PlayerView。创建媒体项(MediaItem),并设置本地视频的URI(对于存储中的文件)或使用rawresourceURI(对于res/raw中的文件)。准备播放器并播放。4.在合适的生命周期方法中释放播放器。我们将分别给出两种方式的代码示例。注意:在AndroidManifest.xml中需要声明读取外部存储的权限(如果视频在外部存储中),并且对于Android6.0以上需要运行时权限。但为了简化,我们假设视频在内部存储中(不需要权限)或者用户已经处理了权限。示例1:播放存储中的视频(内部存储或外部存储,需要文件路径)示例2:播放res/raw目录中的视频开始编写代码示例。</think>以下是使用 Android Media3(ExoPlayer)播放本地视频的完整教程和示例代码: ### 、核心步骤 1. **添加依赖** - 在 `build.gradle` 中添加: ```gradle dependencies { implementation &#39;androidx.media3:media3-exoplayer:1.3.1&#39; implementation &#39;androidx.media3:media3-ui:1.3.1&#39; } ``` 2. **布局文件** (`activity_player.xml`): ```xml <androidx.media3.ui.PlayerView android:id="@+id/player_view" android:layout_width="match_parent" android:layout_height="match_parent" app:show_buffering="when_playing"/> ``` 3. **Activity 代码** (`VideoPlayerActivity.kt`): ```kotlin class VideoPlayerActivity : AppCompatActivity() { private var player: ExoPlayer? = null override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_player) // 1. 创建播放器 player = ExoPlayer.Builder(this).build().apply { // 2. 绑定视图 findViewById<PlayerView>(R.id.player_view).player = this // 3. 设置本地视频URI (示例路径) val videoUri = Uri.parse("file:///sdcard/Movies/sample.mp4") // 替换为实际路径 // 4. 创建媒体项并播放 setMediaItem(MediaItem.fromUri(videoUri)) prepare() play() } } override fun onDestroy() { super.onDestroy() player?.release() // 5. 释放资源 } } ``` ### 二、关键说明 1. **视频路径类型**: - 内部存储:`file:///data/user/0/包名/files/video.mp4` - SD卡路径:`file:///sdcard/Movies/video.mp4`(需权限) - `raw` 资源:`android.resource://${packageName}/${R.raw.video}` 2. **权限处理**(SD卡访问): ```xml <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> ``` 使用 `ActivityResultContracts.RequestPermission()` 请求运行时权限 3. **播放控制**: ```kotlin player_view.findViewById<StyledPlayerControlView>(R.id.exo_controller) ?.showTimeoutMs = 3000 // 控制栏显示时长 ``` ### 三、优化建议 1. **缓冲策略**: ```kotlin player?.setTrackSelectionParameters( player!!.trackSelectionParameters .buildUpon() .setBufferDurationsMs(min = 5000, max = 10000) // 最小/最大缓冲时间 .build() ) ``` 2. **错误处理**: ```kotlin player?.addListener(object : Player.Listener { override fun onPlayerError(error: PlaybackException) { Toast.makeText(this@VideoPlayerActivity, "播放错误: ${error.message}", Toast.LENGTH_SHORT).show() } }) ``` ### 四、完整流程图 ```mermaid graph TD A[添加依赖] --> B[创建PlayerView布局] B --> C[初始化ExoPlayer] C --> D[设置本地视频URI] D --> E[prepare+play] E --> F[自动渲染视频] F --> G[onDestroy时release] ``` > 注意:实际路径需替换为设备中的有效视频文件路径。测试时建议使用 `Context.getFilesDir()` 获取应用专属目录[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值