ข่าวสารเกี่ยวกับผลิตภัณฑ์
ยกระดับการเล่นสื่อ: ขอแนะนำการโหลดล่วงหน้าด้วย Media3 - ตอนที่ 1
ใช้เวลาอ่าน 8 นาที
ในแอปที่เน้นสื่อเป็นหลักในปัจจุบัน การมอบประสบการณ์การเล่นที่ราบรื่นและไม่สะดุดเป็นสิ่งสำคัญต่อประสบการณ์การใช้งานที่ยอดเยี่ยมของผู้ใช้ ผู้ใช้คาดหวังว่าวิดีโอจะเริ่มเล่นทันทีและเล่นได้อย่างราบรื่นโดยไม่มีการหยุดชั่วคราว
ความท้าทายหลักคือเวลาในการตอบสนอง โดยปกติแล้วเพลเยอร์วิดีโอจะเริ่มทำงาน (เชื่อมต่อ ดาวน์โหลด แยกวิเคราะห์ บัฟเฟอร์) หลังจากที่ผู้ใช้เลือกรายการที่จะเล่น แนวทางเชิงรับนี้ช้าเกินไปสำหรับบริบทวิดีโอแบบสั้นในปัจจุบัน วิธีแก้ปัญหาคือการใช้แนวทางเชิงรุก เราต้องคาดการณ์ว่าผู้ใช้จะดูอะไรต่อไปและเตรียมเนื้อหาไว้ล่วงหน้า ซึ่งเป็นหัวใจสำคัญของการโหลดล่วงหน้า
ประโยชน์หลักของการโหลดล่วงหน้ามีดังนี้
- 🚀 วิดีโอเริ่มเล่นเร็วขึ้น: วิดีโอพร้อมเล่นแล้ว ซึ่งจะช่วยให้เปลี่ยนระหว่างรายการต่างๆ ได้เร็วขึ้นและเริ่มเล่นได้ทันทีมากขึ้น
- 📉 บัฟเฟอร์ลดลง: การโหลดข้อมูลล่วงหน้าจะช่วยลดโอกาสที่การเล่นจะหยุดชะงัก เช่น จากปัญหาการเชื่อมต่อเครือข่าย
- ✨ ประสบการณ์ของผู้ใช้ที่ราบรื่นยิ่งขึ้น: การผสมผสานระหว่างการเริ่มเล่นที่เร็วขึ้นและการบัฟเฟอร์ที่น้อยลงจะสร้างการโต้ตอบที่ราบรื่นและต่อเนื่องมากขึ้นสำหรับผู้ใช้
ในซีรีส์ 3 ตอนนี้ เราจะแนะนำและเจาะลึกยูทิลิตีที่มีประสิทธิภาพของ Media3 สำหรับการโหลด (ล่วงหน้า) คอมโพเนนต์
- ในตอนที่ 1 เราจะพูดถึงพื้นฐาน ได้แก่ การทำความเข้าใจกลยุทธ์การโหลดล่วงหน้าที่แตกต่างกันซึ่งมีอยู่ใน Media3, การเปิดใช้ PreloadConfiguration และการตั้งค่า DefaultPreloadManager รวมถึงการเปิดใช้แอปให้โหลดรายการล่วงหน้า เมื่ออ่านบล็อกนี้จบแล้ว คุณจะสามารถโหลดล่วงหน้าและเล่นรายการสื่อด้วยการจัดอันดับและระยะเวลาที่กำหนดค่าไว้
- ใน ตอนที่ 2 เราจะพูดถึงหัวข้อขั้นสูงเพิ่มเติมของ DefaultPreloadManager ได้แก่ การใช้ Listener สำหรับข้อมูลวิเคราะห์, การสำรวจแนวทางปฏิบัติแนะนำที่พร้อมใช้งานจริง เช่น รูปแบบหน้าต่างแบบเลื่อน และคอมโพเนนต์ที่แชร์ที่กำหนดเองของ DefaultPreloadManager และ ExoPlayer
- ในตอนที่ 3 เราจะเจาะลึกการแคชในดิสก์ด้วย DefaultPreloadManager
การโหลดล่วงหน้าช่วยคุณได้ 🦸♀️
แนวคิดหลักเบื้องหลังการโหลดล่วงหน้าคือการโหลดเนื้อหาสื่อก่อนที่คุณจะต้องการ เมื่อผู้ใช้ปัดไปที่วิดีโอถัดไป ระบบจะดาวน์โหลดและพร้อมใช้งานส่วนแรกของวิดีโอแล้ว จึงเล่นได้ทันที
ลองนึกถึงร้านอาหาร ห้องครัวที่วุ่นวายจะไม่รอให้มีออร์เดอร์ก่อนจึงจะเริ่มหั่นหัวหอม 🧅 แต่จะเตรียมส่วนผสมไว้ล่วงหน้า การโหลดล่วงหน้าก็เหมือนการเตรียมส่วนผสมสำหรับเพลเยอร์วิดีโอ
เมื่อเปิดใช้ การโหลดล่วงหน้าจะช่วยลดเวลาในการตอบสนองเมื่อผู้ใช้ข้ามไปยังรายการถัดไปก่อนที่บัฟเฟอร์การเล่นจะไปถึงรายการถัดไป ระบบจะเตรียมช่วงแรกของหน้าต่างถัดไปและบัฟเฟอร์ตัวอย่างวิดีโอ เสียง และข้อความ จากนั้นจะจัดคิวช่วงที่โหลดล่วงหน้าไว้ในเพลเยอร์ โดยมีตัวอย่างที่บัฟเฟอร์ไว้พร้อมใช้งานทันทีและพร้อมที่จะส่งไปยังตัวแปลงสัญญาณสำหรับการแสดงผล
ใน Media3 มี API หลัก 2 รายการสำหรับการโหลดล่วงหน้า ซึ่งแต่ละรายการเหมาะกับ Use Case ที่แตกต่างกัน การเลือก API ที่เหมาะสมเป็นขั้นตอนแรก
1. การโหลดรายการเพลย์ลิสต์ล่วงหน้าด้วย PreloadConfiguration
วิธีนี้เป็นวิธีที่ง่าย เหมาะสำหรับสื่อเชิงเส้นตามลำดับ เช่น เพลย์ลิสต์ที่คาดการณ์ลำดับการเล่นได้ (เช่น ซีรีส์ตอนต่างๆ) คุณให้รายการสื่อทั้งหมดแก่เพลเยอร์โดยใช้ API เพลย์ลิสต์ ของ ExoPlayer และตั้งค่า PreloadConfiguration สำหรับเพลเยอร์ จากนั้นเพลเยอร์จะโหลดรายการถัดไปในลำดับล่วงหน้าโดยอัตโนมัติตามที่กำหนดค่าไว้ API นี้พยายามเพิ่มประสิทธิภาพเวลาในการตอบสนองเมื่อผู้ใช้ข้ามไปยังรายการถัดไปก่อนที่บัฟเฟอร์การเล่นจะไปถึงรายการถัดไป
ระบบจะเริ่มโหลดล่วงหน้าก็ต่อเมื่อไม่มีการโหลดสื่อสำหรับการเล่นที่กำลังดำเนินอยู่ ซึ่งจะป้องกันไม่ให้การโหลดล่วงหน้าแย่งแบนด์วิดท์กับการเล่นหลัก
หากยังไม่แน่ใจว่าคุณต้องโหลดล่วงหน้าหรือไม่ API นี้เป็นตัวเลือกที่ยอดเยี่ยมและง่ายต่อการลองใช้
player.preloadConfiguration =
PreloadConfiguration(/* targetPreloadDurationUs= */ 5_000_000L)
ด้วย PreloadConfiguration ด้านบน เพลเยอร์จะพยายามโหลดสื่อล่วงหน้า 5 วินาทีสำหรับรายการถัดไปในเพลย์ลิสต์
เมื่อเลือกใช้แล้ว คุณสามารถปิดการโหลดเพลย์ลิสต์ล่วงหน้าได้อีกครั้งโดยใช้ PreloadConfiguration.DEFAULT เพื่อปิดใช้การโหลดเพลย์ลิสต์ล่วงหน้า
player.preloadConfiguration = PreloadConfiguration.DEFAULT
2. การโหลดรายการแบบไดนามิกล่วงหน้าด้วย PreloadManager
สำหรับ UI แบบไดนามิก เช่น ฟีดแนวตั้งหรือภาพสไลด์ที่รายการ "ถัดไป" กำหนดโดยการโต้ตอบของผู้ใช้ API PreloadManager เหมาะสมกว่า ซึ่งเป็นคอมโพเนนต์ใหม่ที่มีประสิทธิภาพและทำงานแบบสแตนด์อโลนภายในไลบรารี Media3 ExoPlayer ที่ออกแบบมาโดยเฉพาะเพื่อโหลดล่วงหน้า โดยจะจัดการคอลเล็กชัน MediaSource ที่เป็นไปได้ โดยจัดลำดับความสำคัญตามความใกล้เคียงกับตำแหน่งปัจจุบันของผู้ใช้ และให้การควบคุมแบบละเอียด เกี่ยวกับสิ่งที่ต้องโหลดล่วงหน้า เหมาะสำหรับสถานการณ์ที่ซับซ้อน เช่น ฟีดวิดีโอแบบสั้นแบบไดนามิก
การตั้งค่า PreloadManager
DefaultPreloadManager เป็นการใช้งานมาตรฐานสำหรับ PreloadManager
ตัวสร้าง DefaultPreloadManager สามารถสร้างทั้ง DefaultPreloadManager และอินสแตนซ์ ExoPlayer ที่จะเล่นเนื้อหาที่โหลดล่วงหน้า หากต้องการสร้าง DefaultPreloadManager คุณจะต้องส่ง TargetPreloadStatusControl ซึ่งเครื่องมือจัดการการโหลดล่วงหน้าจะใช้เพื่อค้นหาปริมาณที่จะโหลดสำหรับรายการ เราจะอธิบายและกำหนดตัวอย่าง TargetPreloadStatusControl ในส่วนด้านล่าง
val preloadManagerBuilder = DefaultPreloadManager.Builder(context, targetPreloadStatusControl) val preloadManager = val preloadManagerBuilder.build() // Build ExoPlayer with DefaultPreloadManager.Builder val player = preloadManagerBuilder.buildExoPlayer()
คุณต้องใช้ เครื่องมือสร้าง เดียวกันสำหรับทั้ง ExoPlayer และ DefaultPreloadManager เพื่อให้มั่นใจว่าคอมโพเนนต์เบื้องหลังจะแชร์กันอย่างถูกต้อง
เท่านี้ก็เรียบร้อย ตอนนี้คุณมีเครื่องมือจัดการที่พร้อมรับคำสั่งแล้ว
การกำหนดค่าระยะเวลาและการจัดอันดับด้วย TargetPreloadStatusControl
จะเกิดอะไรขึ้นหากคุณต้องการโหลดวิดีโอล่วงหน้า 10 วินาที คุณระบุตำแหน่งของรายการสื่อในภาพสไลด์ได้ และ DefaultPreloadManager จะจัดลำดับความสำคัญในการโหลดรายการตามความใกล้เคียงกับรายการที่ผู้ใช้กำลังเล่น
หากต้องการควบคุมระยะเวลาของรายการที่จะโหลดล่วงหน้า คุณสามารถระบุได้ด้วย DefaultPreloadManager.PreloadStatus ที่คุณส่งคืน
ตัวอย่างเช่น
- รายการ "A" มีลำดับความสำคัญสูงสุด ให้โหลดวิดีโอล่วงหน้า 5 วินาที
- รายการ "B" มีลำดับความสำคัญปานกลาง แต่เมื่อถึงรายการนี้ ให้โหลดวิดีโอล่วงหน้า 3 วินาที
- รายการ "C" มีลำดับความสำคัญต่ำ ให้โหลดเฉพาะแทร็ก
- รายการ "D" มีลำดับความสำคัญต่ำกว่า ให้เตรียมไว้เท่านั้น
- รายการอื่นๆ อยู่ไกล ไม่ต้องโหลดล่วงหน้า
การควบคุมแบบละเอียดนี้ช่วยให้คุณเพิ่มประสิทธิภาพการใช้ทรัพยากร ซึ่งแนะนำให้ทำเพื่อการเล่นที่ราบรื่น
import androidx.media3.exoplayer.DefaultPreloadManager.PreloadStatus
class MyTargetPreloadStatusControl(
currentPlayingIndex: Int = C.INDEX_UNSET
) : TargetPreloadStatusControl<Int,PreloadStatus> {
// The app is responsible for updating this based on UI state
override fun getTargetPreloadStatus(index: Int): PreloadStatus? {
val distance = index - currentPlayingIndex
// Adjacent items (Next): preload 5 seconds
if (distance == 1) {
// Return a PreloadStatus that is labelled by STAGE_SPECIFIED_RANGE_LOADED and suggest loading // 5000ms from the default start position
return PreloadStatus.specifiedRangeLoaded(5000L)
}
// Adjacent items (Previous): preload 3 seconds
else if (distance == -1) {
// Return a PreloadStatus that is labelled by STAGE_SPECIFIED_RANGE_LOADED //and suggest loading 3000ms from the default start position
return PreloadStatus.specifiedRangeLoaded(3000L)
}
// Items two positions away: just select tracks
else if (distance) == 2) {
// Return a PreloadStatus that is labelled by STAGE_TRACKS_SELECTED
return PreloadStatus.TRACKS_SELECTED
}
// Items four positions away: just select prepare
else if (abs(distance) <= 4) {
// Return a PreloadStatus that is labelled by STAGE_SOURCE_PREPARED
return PreloadStatus.SOURCE_PREPARED
}
// All other items are too far away
return null
}
}
เคล็ดลับ: PreloadManager สามารถโหลดทั้งรายการก่อนหน้าและรายการถัดไปล่วงหน้าได้ ในขณะที่ PreloadConfiguration จะดูเฉพาะรายการถัดไปเท่านั้น
การจัดการรายการที่โหลดล่วงหน้า
เมื่อสร้างเครื่องมือจัดการแล้ว คุณจะเริ่มบอกเครื่องมือจัดการได้ว่าจะให้ทำอะไร เมื่อผู้ใช้เลื่อนดูฟีด คุณจะระบุวิดีโอที่จะมาถึงและเพิ่มวิดีโอเหล่านั้นลงในเครื่องมือจัดการ การโต้ตอบกับ PreloadManager เป็นการสนทนาที่ขับเคลื่อนด้วยสถานะระหว่าง UI กับเครื่องมือโหลดล่วงหน้า
1. เพิ่มรายการสื่อ
เมื่อป้อนข้อมูลในฟีด คุณต้องแจ้งให้ เครื่องมือจัดการ ทราบเกี่ยวกับสื่อที่ต้องติดตาม หากเพิ่งเริ่มต้น คุณสามารถเพิ่มรายการทั้งหมดที่ต้องการโหลดล่วงหน้าได้ จากนั้นคุณจะเพิ่มรายการเดียวลงในรายการได้ตามต้องการ คุณสามารถควบคุมรายการที่อยู่ในรายการโหลดล่วงหน้าได้อย่างเต็มที่ ซึ่งหมายความว่าคุณต้องจัดการสิ่งที่เพิ่มและนำออกจากเครื่องมือจัดการด้วย
val initialMediaItems = pullMediaItemsFromService(/* count= */ 20)
for (index in 0 until initialMediaItems.size) {
preloadManager.add(
initialMediaItems.get(index),index)
)
}
ตอนนี้เครื่องมือจัดการจะเริ่มดึงข้อมูลสำหรับ MediaItem นี้ในเบื้องหลัง
หลังจากเพิ่มแล้ว ให้บอกเครื่องมือจัดการให้ประเมินรายการใหม่ (บอกเป็นนัยว่ามีการเปลี่ยนแปลง เช่น การเพิ่ม/ นำรายการออก หรือผู้ใช้เปลี่ยนไปเล่นรายการใหม่)
preloadManager.invalidate()
2. ดึงและเล่นรายการ
นี่คือตรรกะการเล่นหลัก เมื่อผู้ใช้ตัดสินใจเล่นวิดีโอนั้น คุณไม่จำเป็นต้องสร้าง MediaSource ใหม่ แต่ให้ขอ PreloadManager สำหรับรายการที่เตรียมไว้แล้ว คุณสามารถดึง MediaSource จาก PreloadManager โดยใช้ MediaItem
หากรายการที่ดึงมาจาก PreloadManager เป็น Null แสดงว่า mediaItem ยังไม่ได้โหลดล่วงหน้าหรือเพิ่มลงใน PreloadMamager ดังนั้นคุณจึงเลือกที่จะตั้งค่า mediaItem โดยตรง
// When a media item is about to display on the screen
val mediaSource = preloadManager.getMediaSource(mediaItem)
if (mediaSource!= null) {
player.setMediaSource(mediaSource)
} else {
// If mediaSource is null, that mediaItem hasn't been added yet.
// So, send it directly to the player.
player.setMediaItem(mediaItem)
}
player.prepare()
// When the media item is displaying at the center of the screen
player.play()
การเตรียม MediaSource ที่ดึงมาจาก PreloadManager จะช่วยให้คุณเปลี่ยนจากการโหลดล่วงหน้าเป็นการเล่นได้อย่างราบรื่น โดยใช้ข้อมูลที่อยู่ในหน่วยความจำอยู่แล้ว ซึ่งจะทำให้เวลาเริ่มต้นเร็วขึ้น
3. ซิงค์ดัชนีปัจจุบันกับ UI
เนื่องจากฟีด / รายการของเราอาจเป็นแบบไดนามิก จึงควรแจ้งให้ PreloadManager ทราบดัชนีการเล่นปัจจุบันเพื่อให้เครื่องมือจัดการจัดลำดับความสำคัญในการโหลดล่วงหน้าสำหรับรายการที่อยู่ใกล้กับดัชนีปัจจุบันมากที่สุด
preloadManager.setCurrentPlayingIndex(currentIndex) // Need to call invalidate() to update the priorities preloadManager.invalidate()
4. นำรายการออก
คุณควรนำรายการที่เครื่องมือจัดการไม่จำเป็นต้องติดตามออก เช่น รายการที่อยู่ไกลจากตำแหน่งปัจจุบันของผู้ใช้ เพื่อให้เครื่องมือจัดการทำงานได้อย่างมีประสิทธิภาพ
// When an item is too far from the current playing index preloadManager.remove(mediaItem)
หากต้องการล้างรายการทั้งหมดพร้อมกัน ให้เรียกใช้ preloadManager.reset()
5. ปล่อยเครื่องมือจัดการ
เมื่อไม่ต้องการใช้ PreloadManager อีกต่อไป (เช่น เมื่อ UI ถูกทำลาย) คุณต้องปล่อยเครื่องมือจัดการเพื่อเพิ่มพื้นที่ว่างให้กับทรัพยากร ตำแหน่งที่เหมาะสมในการดำเนินการนี้คือตำแหน่งที่คุณปล่อยทรัพยากรของเพลเยอร์อยู่แล้ว เราขอแนะนำให้ปล่อยเครื่องมือจัดการก่อนเพลเยอร์ เนื่องจากเพลเยอร์จะเล่นต่อไปได้หากคุณไม่ต้องการโหลดล่วงหน้าอีก
// In your Activity's onDestroy() or Composable's onDispose preloadManager.release()
ช่วงเวลาสาธิตการใช้งาน
ดูการใช้งานจริงแบบสดๆ 👍
ในการสาธิตด้านล่าง เราจะเห็นผลกระทบของ PreloadManager ทางด้านขวา ซึ่งมีเวลาโหลดที่เร็วกว่า ในขณะที่ด้านซ้ายแสดงประสบการณ์การใช้งานที่มีอยู่ นอกจากนี้ คุณยังดูตัวอย่างโค้ด sample สำหรับการสาธิตได้ด้วย (โบนัส: ระบบจะแสดงเวลาในการตอบสนองเมื่อเริ่มต้นสำหรับวิดีโอทุกรายการด้วย)
ขั้นตอนต่อไปคือ
จบตอนที่ 1 แล้ว ตอนนี้คุณมีเครื่องมือในการสร้างระบบการโหลดล่วงหน้าแบบไดนามิกแล้ว คุณสามารถใช้ PreloadConfiguration เพื่อโหลดรายการถัดไปของเพลย์ลิสต์ล่วงหน้าใน ExoPlayer หรือตั้งค่า DefaultPreloadManager เพิ่มและนำรายการออกได้ทันที กำหนดค่าสถานะการโหลดล่วงหน้าเป้าหมาย และดึงเนื้อหาที่โหลดล่วงหน้ามาเล่นได้อย่างถูกต้อง
ใน ตอนที่ 2 เราจะเจาะลึก DefaultPreloadManager เราจะสำรวจวิธีฟังเหตุการณ์การโหลดล่วงหน้า พูดคุยเกี่ยวกับแนวทางปฏิบัติแนะนำ เช่น การใช้หน้าต่างแบบเลื่อนเพื่อหลีกเลี่ยงปัญหาหน่วยความจำ และดูคอมโพเนนต์ที่แชร์ที่กำหนดเองของ ExoPlayer และ DefaultPreloadManager
คุณมีความคิดเห็นที่ต้องการแชร์ไหม เรายินดีรับฟังความคิดเห็นจากคุณ
โปรดติดตามและไปทำให้แอปของคุณเร็วขึ้น 🚀
อ่านต่อ
-
ข่าวสารเกี่ยวกับผลิตภัณฑ์
ยินดีต้อนรับสู่ตอนที่ 2 ของซีรีส์ 3 ตอนเกี่ยวกับการโหลดสื่อล่วงหน้าด้วย Media3 ซีรีส์นี้ออกแบบมาเพื่อแนะนำคุณตลอดกระบวนการสร้างประสบการณ์การใช้งานสื่อที่ตอบสนองสูงและมีเวลาในการตอบสนองต่ำในแอป Android
Mayuri Khinvasara Khabya • ใช้เวลาอ่าน 9 นาที
-
ข่าวสารเกี่ยวกับผลิตภัณฑ์
ระบบนิเวศมือถือมีการพัฒนาอยู่เสมอ ซึ่งนำมาซึ่งโอกาสใหม่ๆ และภัยคุกคามใหม่ๆ แม้จะมีการเปลี่ยนแปลงเหล่านี้ แต่ Android และ Google Play ยังคงมุ่งมั่นที่จะทำให้ผู้ใช้หลายพันล้านคนสามารถเพลิดเพลินกับแอปได้อย่างมั่นใจ และนักพัฒนาแอปสามารถสร้างสรรค์นวัตกรรมได้อย่างเต็มที่
Vijaya Kaza • ใช้เวลาอ่าน 3 นาที
-
ข่าวสารเกี่ยวกับผลิตภัณฑ์
Jetpack Compose รุ่นเมษายน '26 พร้อมให้ใช้งานอย่างเสถียรแล้ว รุ่นนี้มีโมดูลหลักของ Compose เวอร์ชัน 1.11 (ดูการแมป BOM แบบเต็ม), เครื่องมือแก้ไขข้อบกพร่องขององค์ประกอบที่แชร์, เหตุการณ์แทร็กแพด และอื่นๆ
Meghan Mehta • ใช้เวลาอ่าน 5 นาที
รับข่าวสาร
รับข้อมูลเชิงลึกล่าสุดเกี่ยวกับการพัฒนา Android ส่งตรงถึงกล่องจดหมายของคุณ ทุกสัปดาห์