HarmonyOS 鸿蒙Next中使用AVPlayer播放音频后切换音频文件报错
HarmonyOS 鸿蒙Next中使用AVPlayer播放音频后切换音频文件报错
AVPlayer.reset前加个await就行了
export class AudioPlayUtils {
private avPlayer: media.AVPlayer | null = null;
constructor() {
media.createAVPlayer().then(async (avPlayer: media.AVPlayer) => {
this.avPlayer = avPlayer;
this.playerCallback(avPlayer);
})
}
async audioPrepare(mediaUrl: string) {
let fileDescriptor = await getContext().resourceManager.getRawFd(mediaUrl);
let avFileDescriptor: media.AVFileDescriptor =
{ fd: fileDescriptor.fd, offset: fileDescriptor.offset, length: fileDescriptor.length };
if (this.avPlayer) {
this.avPlayer.stop()
this.avPlayer.reset()
this.avPlayer.fdSrc = avFileDescriptor
}
}
audioPlay() {
this.avPlayer?.play()
}
audioPause() {
this.avPlayer?.pause()
}
release() {
this.avPlayer?.release()
}
// 注册avplayer回调函数
playerCallback(avPlayer: media.AVPlayer) {
avPlayer.on('timeUpdate', (time: number) => {
// console.info(TAG, 'AVPlayer timeUpdate. time = ' + time / 1000);
})
// error回调监听函数,当avPlayer在操作过程中出现错误时调用 reset接口触发重置流程
avPlayer.on('error', (err: BusinessError) => {
console.error(TAG, `Invoke avPlayer failed, code is ${err.code}, message is ${err.message}`);
avPlayer.reset(); // 调用reset重置资源,触发idle状态
})
// 状态机变化回调函数
avPlayer.on('stateChange', async (state: string, reason: media.StateChangeReason) => {
switch (state) {
case 'idle': // 成功调用reset接口后触发该状态机上报
console.info(TAG, 'AVPlayer state idle called.');
break;
case 'initialized': // avplayer 设置播放源后触发该状态上报
console.info(TAG, 'AVPlayer state initialized called.');
if (this.avPlayer) {
avPlayer.prepare();
}
break;
case 'prepared': // prepare调用成功后上报该状态机
console.info(TAG, 'AVPlayer state prepared called.');
avPlayer.play()
break;
case 'completed': // prepare调用成功后上报该状态机
console.info(TAG, 'AVPlayer state completed called.');
avPlayer.stop();
break;
case 'playing': // play成功调用后触发该状态机上报
console.info(TAG, 'AVPlayer state playing called.');
break;
case 'paused': // pause成功调用后触发该状态机上报
console.info(TAG, 'AVPlayer state paused called.');
break;
case 'stopped': // stop接口成功调用后触发该状态机上报
console.info(TAG, 'AVPlayer state stopped called.');
avPlayer.reset();
break;
case 'released':
console.info(TAG, 'AVPlayer state released called.');
break;
default:
console.info(TAG, 'AVPlayer state unknown called.');
break;
}
})
}
}
调用处
playUtils: AudioPlayUtils = new AudioPlayUtils()
this.playUtils.audioPrepare(rawFile文件)
我想使用一个avPlay对象,使用audioPrepare切换其他音频源,而不用销毁avPlay重新创建,但是会报错5400102或者401
更多关于HarmonyOS 鸿蒙Next中使用AVPlayer播放音频后切换音频文件报错的实战教程也可以访问 https://www.itying.com/category-93-b0.html
2 回复
鸿蒙Next中AVPlayer切换音频文件报错可能是未正确处理资源释放和重新初始化流程。需在切换前调用reset()或release()方法释放当前播放实例,然后重新创建AVPlayer对象并设置新的数据源。确保在prepareAsync()完成后再执行play()操作。常见错误代码201可能表示无效的操作状态。检查音频文件路径是否正确及格式是否支持(如AAC、MP3等)。
更多关于HarmonyOS 鸿蒙Next中使用AVPlayer播放音频后切换音频文件报错的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html
在HarmonyOS Next中使用AVPlayer切换音频源时出现5400102或401错误,通常是由于状态机管理不当导致的。以下是关键点分析:
- 主要问题在于
reset()
操作是异步的,但代码中直接同步调用了后续操作。正确的做法是在audioPrepare
方法中为reset()
添加await
:
async audioPrepare(mediaUrl: string) {
let fileDescriptor = await getContext().resourceManager.getRawFd(mediaUrl);
let avFileDescriptor: media.AVFileDescriptor =
{ fd: fileDescriptor.fd, offset: fileDescriptor.offset, length: fileDescriptor.length };
if (this.avPlayer) {
await this.avPlayer.stop();
await this.avPlayer.reset(); // 关键修改点
this.avPlayer.fdSrc = avFileDescriptor;
}
}
- 错误码说明:
- 5400102:表示在非法状态下执行了操作
- 401:表示参数无效或操作不被允许
- 状态机注意事项:
- 必须在idle状态下才能设置新的播放源(
fdSrc
) reset()
会将播放器状态重置为idle,但需要等待操作完成- 建议在
stateChange
回调的idle状态中设置新源,这样更可靠
- 其他建议修改点:
- 在
stateChange
回调中,completed
状态直接调用reset()
可能会造成循环,建议移除 - 确保每次操作前检查
avPlayer
是否为null
这些修改应该能解决切换音频源时的报错问题。