(已解决)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()释放资源,再设置新数据源。确保在onStateChange回调中处理PREPARED状态后再调用play()

关键代码示例:

avPlayer.reset();
avPlayer.setSource(/*新音频URI*/);
avPlayer.prepare();

常见错误是未等待PREPARED状态直接播放,或未正确处理前一个音频实例。检查媒体源格式是否支持,路径是否正确。网络流需确保URL有效。

更多关于(已解决)HarmonyOS 鸿蒙Next中使用AVPlayer播放音频后,切换音频文件播放报错的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html


这是一个典型的AVPlayer状态管理问题。在HarmonyOS Next中,AVPlayer切换音频源时需要正确处理状态转换流程。根据您提供的代码,主要问题在于没有等待reset操作完成就立即设置新的播放源。

解决方案要点:

  1. reset()是异步操作,需要添加await确保完成
  2. 正确的播放源切换流程应该是:stop -> await reset -> 设置新源 -> prepare -> play

修改后的关键代码段应该是:

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();
    await this.avPlayer.reset(); // 关键修改:添加await
    this.avPlayer.fdSrc = avFileDescriptor;
    this.avPlayer.prepare(); // 需要重新prepare
  }
}

错误码解释:

  • 5400102:通常表示在错误的状态下执行操作
  • 401:参数错误,可能由于状态未正确重置导致

这个修改确保了状态转换的完整性,避免了在错误状态下设置播放源。

回到顶部