基于鸿蒙API10的RTSP播放器(一:基本界面的实现)

news2025/1/11 2:52:26

ijkplayer简介:
ijkplayer 本身是一个开源的 Android 媒体播放库,它主要用于播放视频和音频文件,现在已有前辈将其引入到鸿蒙当中,通过XComponent组件完成适配。向开源致敬!

支持格式:
它支持多种格式,包括 MP4、M3U8、FLV 等,并且具有高性能和低功耗的特点。ijkplayer 基于 FFmpeg 进行开发,通过 JNI 技术将 FFmpeg 的功能封装在 Java 代码中,从而实现了对 FFmpeg 的高效调用。

ijkplayer 的关键特点:

  1. 高性能ijkplayer 利用 FFmpeg 的强大功能,能够快速、高效地播放视频和音频文件。
  2. 低功耗:与一些其他播放器相比,ijkplayer 在播放过程中消耗的电量较少,更适合长时间使用的场景。
  3. 丰富的格式支持ijkplayer 支持多种视频和音频格式,包括但不限于 MP4、M3U8、FLV 等。
  4. 自定义配置:开发者可以根据需要自定义播放器的各种参数,如音量、播放速度、视频宽高比等。
  5. 丰富的监听器ijkplayer 提供了一系列的监听器,如 OnPreparedListener、OnCompletionListener 等,方便开发者监听播放过程中的各种事件。
  6. 支持硬解码:在一些硬件设备上,ijkplayer 能够利用硬件加速器进行视频解码,从而提高播放效率。
  7. 开源免费ijkplayer 是一个开源项目,可以在 GitHub 上找到其源代码,并且免费使用。

鸿蒙版ijkplayer
已经更新到2.0.4,在鸿蒙三方库中心仓库可以找到,在DevEco终端中,可以通过命令 ohpm install @ijkplayer 直接下载。鸿蒙版ijkplayer2.0.3-rc.2已经可以支持X86架构的模拟器,并且,2.0.3-rc.3可以支持H265硬编码,以及直播流RTSP。
在这里插入图片描述

鸿蒙三方库中心仓地址:鸿蒙版ijkplayer地址

修改API版本:
API10的鸿蒙项目文件中,在引入鸿蒙版ijkplayer后,修改三方库当中的src中的module.json文件,将API 12 改成 API 10 。其中依赖引入有一些小bug,会随机报错,直接忽略。
在这里插入图片描述

运行结果:
在这里插入图片描述

页面代码:


// 1.这个页面仅用于ijkplayer测试运行
// 2.编写这个页面进行测试,包括画面 声音 颜色的测试,也包括暂停、开始、返回的测试
// 3.这是一个简单的页面,如需复杂功能,比如添加视频播放轨道,参见三方库中心仓的API说明
// 4.致敬开源

import {  
  IjkMediaPlayer,  
  OnErrorListener,  
  OnInfoListener,  
  OnSeekCompleteListener } from "@ohos/ijkplayer";  
import type { OnPreparedListener } from "@ohos/ijkplayer";  
import type { OnVideoSizeChangedListener } from "@ohos/ijkplayer";  
import { LogUtils } from "@ohos/ijkplayer";  
import { promptAction, router } from '@kit.ArkUI';  
  
  
// 获取ijkplayer实例  
let mIjkMediaPlayer = IjkMediaPlayer.getInstance();  
  
@Entry  
@Component  
struct Index {  
  // 苹果视频流  
  @State private videoUrl: string = "http://devimages.apple.com.edgekey.net/streaming/examples/bipbop_4x3/bipbop_4x3_variant.m3u8";  
  @State private aspRatio: number = 16 / 9; // 默认宽高比  
  // XComponent的上下文  
  @State mContext: object | undefined = undefined;  
  
  
  build() {  
    Column() {  
      // 视频播放组件  
      Row() {  
        XComponent({  
          id: 'xcomponentId',  
          type: 'surface',  
          libraryname: 'ijkplayer_napi'  
        })  
          .onLoad((context?: object) => {  
            if (!!context) {  
              this.mContext = context;  
              this.startPlay();  
            }  
          })  
          .onDestroy(() => {  
            this.releaseMediaPlayer();  
          })  
          .width('100%')  
      }  
      .height("900px")  
      .width('100%')  
  
      // 控制按钮组件  
      Row({space:"60px"}) {  
        Button( 'Play')  
          .width('20%')  
          .height('50vp')  
          .onClick(() => {  
              this.restartPlay();  
          });  
        Button('Pause')  
          .width('20%')  
          .height('50vp')  
          .onClick(() => {  
              this.pause();  
          });  
        Button('Back')  
          .width('20%')  
          .height('50vp')  
          .onClick(() => {  
            router.back();  
          })  
      }  
      .width('100%')  
      .justifyContent(FlexAlign.Center)  
      .margin("50px")  
    }  
    .height('100%')  
  }  
  
  // 暂停后恢复播放  
  restartPlay() {  
    mIjkMediaPlayer.start();  
  }  
  // 退出XComponent后释放资源,避免退出组件后仍然在持续接受流的情况  
  releaseMediaPlayer() {  
    mIjkMediaPlayer.stop();  
    mIjkMediaPlayer.release();  
  }  
  // 暂停  
  pause() {  
    mIjkMediaPlayer.pause();  
  }  
  
  
  startPlay() {  
      this.play(this.videoUrl);// 调用play方法播放视频  
  }  
  
  // 播放方法,调用ijkplayer库中已经写好的方法  
  private play(url: string) {  
  
    let that = this  
  
    // 以下内容为配置mIjkplayer对象的属性!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!  
  
    // 设置XComponent回调的context  
    mIjkMediaPlayer.setContext(this.mContext);  
    //初始化配置  
    mIjkMediaPlayer.native_setup();  
    // 设置debug模式,可以根基自己需要,调整成true,会打印更多内容便于调试  
	mIjkMediaPlayer.setDebug(false);
    //设置视频源  
    mIjkMediaPlayer.setDataSource(url);  
    //设置视频源http请求头  
    let headers = new Map([  
      ["user_agent", "Mozilla/5.0 BiliDroid/7.30.0 (bbcallen@gmail.com)"],  
      ["referer", "https://www.bilibili.com"]  
    ]);  
    mIjkMediaPlayer.setDataSourceHeader(headers);  
    //使用精确寻帧 例如,拖动播放后,会寻找最近的关键帧进行播放,很有可能关键帧的位置不是拖动后的位置,而是较前的位置.可以设置这个参数来解决问题  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "enable-accurate-seek", "1");  
    //预读数据的缓冲区大小  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "max-buffer-size", "102400");  
    //停止预读的最小帧数  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "min-frames", "100");  
    //启动预加载  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "start-on-prepared", "1");  
    // 设置无缓冲,这是播放器的缓冲区,有数据就播放  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "packet-buffering", "0");  
    //跳帧处理,放CPU处理较慢时,进行跳帧处理,保证播放流程,画面和声音同步  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "framedrop", "5");  
    // 最大缓冲cache是3s, 有时候网络波动,会突然在短时间内收到好几秒的数据  
    // 因此需要播放器丢包,才不会累积延时  
    // 这个和第三个参数packet-buffering无关。  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "max_cached_duration", "3000");  
    // 无限制收流  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "infbuf", "1");  
    mIjkMediaPlayer.setOptionLong(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "infbuf", "1")  
    // 屏幕常亮  
    mIjkMediaPlayer.setScreenOnWhilePlaying(true);  
    // 设置超时  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_FORMAT, "timeout", "10000000");  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_FORMAT, "connect_timeout", "10000000");  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_FORMAT, "addrinfo_timeout", "10000000");  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_FORMAT, "dns_cache_timeout", "10000000");  
    // 设置音量  
    mIjkMediaPlayer.setVolume("0.5", "0.5");  
    // 变速播放  
    mIjkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "soundtouch", "1");  
    // 获取并打印播放速度  
    let Speed = mIjkMediaPlayer.getSpeed()  
    LogUtils.getInstance().LOGI('getSpeed--' + Speed)  
    //是否开启循环播放  
    mIjkMediaPlayer.setLoopCount(true);  
  
    // 以下内容为配置mIjkplayer对象的监听器!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!  
  
    // 设置视频尺寸变化的监听器  
    let mOnVideoSizeChangedListener: OnVideoSizeChangedListener = {  
      onVideoSizeChanged: (width: number, height: number, sar_num: number, sar_den: number) => {  
        that.aspRatio = width / height;  
        LogUtils.getInstance()  
          .LOGI("setOnVideoSizeChangedListener-->go:" + width + "," + height + "," + sar_num + "," + sar_den)  
      }  
    }    mIjkMediaPlayer.setOnVideoSizeChangedListener(mOnVideoSizeChangedListener);  
  
    // 设置视频准备完成的监听器  
    let mOnPreparedListener: OnPreparedListener = {  
      onPrepared: () => {  
        LogUtils.getInstance().LOGI("setOnPreparedListener-->go");  
      }  
    }    mIjkMediaPlayer.setOnPreparedListener(mOnPreparedListener);  
  
  
    // 设置拖动播放完成的监听器  
    let mOnSeekCompleteListener: OnSeekCompleteListener = {  
      onSeekComplete: () => {  
        LogUtils.getInstance().LOGI("OnSeekCompleteListener-->go");  
        that.startPlay();  
      }  
    }    mIjkMediaPlayer.setOnSeekCompleteListener(mOnSeekCompleteListener);  
  
    // 设置信息监听器  
    let mOnInfoListener: OnInfoListener = {  
      onInfo: (what: number, extra: number) => {  
        LogUtils.getInstance().LOGI("OnInfoListener-->go:" + what + "===" + extra);  
      }  
    }    mIjkMediaPlayer.setOnInfoListener(mOnInfoListener);  
  
    // 设置错误监听器  
    let mOnErrorListener: OnErrorListener = {  
      onError: (what: number, extra: number) => {  
        LogUtils.getInstance().LOGI("OnErrorListener-->go:" + what + "===" + extra)  
        promptAction.showToast({  
          message: "视频播放异常"  
        });  
      }  
    }    mIjkMediaPlayer.setOnErrorListener(mOnErrorListener);  
  
    // 设置消息监听器  
    mIjkMediaPlayer.setMessageListener();  
    // 加载视频  
    mIjkMediaPlayer.prepareAsync();  
    // 开始播放(真正播放的入口,通过NAPI调用底层)  
    mIjkMediaPlayer.start();  
  }  
  
}

关闭mijkplayer对象的debug后,会继续打印以下日志:

根据您提供的日志信息,这些日志似乎是来自一个应用程序的错误和性能监控输出。下面是对这些日志的逐条解释:

  1. 在这里插入图片描述

    • 这条日志表明在处理名为 com.example.app.forrtsp 的应用程序时,出现了 getaddrinfo_ext 函数调用失败的情况。错误代码 218 通常与DNS查询失败有关,而 -2 可能是表示一个具体的错误状态。可能意味着应用程序无法解析视频播放地址的DNS。
  2. 在这里插入图片描述

    • 这条日志是关于EGL(Embedded System Grapfics Library嵌入式图形库)的输出缓冲区交换操作。d_eglSwapBuffers_special 是EGL的一个函数,它负责在OpenGL ES上下文中交换渲染缓冲区。thread output size 14420760 表示线程输出的缓冲区大小,而 speed 14028k/s 表示每秒交换的次数。API_num 450 可能是表示使用的是OpenGL ES 3.0。
  3. 在这里插入图片描述

    • 这条日志来自应用程序的渲染器部分,可能是在尝试写入缓存数据时状态不正确。这可能是一个同步问题或者状态管理错误。
    • WriteInner Status changed while write这条日志也是关于渲染器的,可能是在写入操作期间状态发生了变化。

EGL(嵌入式图形库)
是OpenGL ES的一部分,用于管理图形上下文和表面。提供了一个与图形硬件和窗口系统无关的接口,允许开发者在多种平台上编写图形应用程序。EGL的主要功能:

  1. 创建和管理图形上下文:EGL提供了一种机制,可以在多种图形硬件和窗口系统上创建和管理OpenGL ES或OpenVG上下文。
  2. 管理表面:EGL可以创建和管理不同类型的图形表面,如窗口、PBuffer(内存中的表面)和帧缓冲区。
  3. 同步机制:EGL提供了同步机制,如等待交换完成和同步上下文,以避免在渲染过程中出现竞态条件。
  4. 加载OpenGL ES或OpenVG函数:EGL可以加载和卸载OpenGL ES或OpenVG函数,以便在不同的上下文中使用。
  5. 缓冲区交换:EGL提供了缓冲区交换功能,允许开发者控制何时将渲染结果呈现到屏幕上。
    在Android和鸿蒙等嵌入式平台上,EGL是图形开发中不可或缺的一部分,它为开发者提供了一个跨平台的图形接口,使得在不同的硬件和操作系统上编写图形应用程序变得更加容易。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2131587.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

wine 麒麟系统运行Windows编译的exe

启动终端 首先,我们需要打开终端程序。 在终端中,输入以下命令: sudo apt-get update这个命令通常会要求输入密码,输入你的密码后,就可以继续执行了。 如果你想要切换到 root 用户,可以通过以下方法: su 用户名然后输入 root 用户的密码。 如果你想要重置 root 用户…

网站如何防范BOT流量?

随着互联网的快速发展,BOT流量已成为网络安全领域的一大隐患。BOT,即自动化程序,它们在网络中执行各种任务,包括数据抓取、恶意抢购、暴力破解等。这些行为不仅威胁到网站的正常运营,还可能造成数据泄露、经济损失等严…

day20JS-axios数据通信

1. 什么是axios axios 是一个基于Promise 用于浏览器和 nodejs 的 HTTP 客户端,简单的理解就是ajax的封装,只不过它是Promise的实现版本。 特性: 从浏览器中创建 XMLHttpRequests从 node.js 创建 http 请求支持 Promise API拦截请求和响应转…

零宽字符应用场景及前端解决方案

零宽字符(Zero Width Characters)是一类在文本中不可见但具有特定功能的特殊字符。称为零宽字符,也叫幽灵字符。它们在显示时不占据任何空间,但在文本处理和显示中发挥着重要作用。这些字符主要包括零宽度空格、零宽度非连接符、零…

【C++ 高频面试题】STL 你了解多少呢?vector 的底层实现原理

文章目录 1. 常见的 STL 容器2.vector 和 list 的区别3. vector 的底层原理4. push_back() 和 emplace_back() 区别 1. 常见的 STL 容器 🍎①序列容器 vector(向量):是一个动态数组实现,提供高效的随机访问和在尾部进行…

基于R语言结构方程模型分析与实践技术应用

结构方程模型(Sructural Equation Model)是一种建立、估计和检验研究系统中多变量间因果关系的模型方法,它可以替代多元回归、因子分析、协方差分析等方法,利用图形化模型方式清晰展示研究系统中变量间的因果网络关系,…

信息安全数学基础(9)素数的算数基本定理

前言 在信息安全数学基础中,素数的算数基本定理(也称为唯一分解定理或算术基本定理)是一个极其重要的定理,它描述了正整数如何唯一地分解为素数的乘积。这个定理不仅是数论的基础,也是许多密码学算法(如RSA…

【算法专场】分治(上)

目录 前言 什么是分治? 75. 颜色分类 算法分析 算法步骤 算法代码 912. 排序数组 - 力扣(LeetCode) 算法分析 算法步骤 算法代码 215. 数组中的第K个最大元素 - 力扣(LeetCode) 算法分析 算法步骤 ​编辑…

【Redis入门到精通一】什么是Redis?

目录 Redis 1. Redis的背景知识 2.Redis特性 3.Redis的使用场景 4.Ubuntu上安装配置Redis Redis Redis在当今编程技术中的地位可以说非常重要,大多数互联网公司内部都在使用这个技术,熟练使用Redis已经成为开发人员的一个必备技能。本章将带领读者…

yolov5实战全部流程

本科生阶段除了在中等以上的985和某些特定的CS强项院校,无意愿研究生学习的本科生是不建议学习人工智能这一专业的,保研学生也许可以在实验室打工推荐信学习接触到此类事件,此项blog主要是对yolov5的实践性项目,yolov5作为最具有代…

python 函数 封装

封装 函数的参数是:变量 def 函数(参数):print(参数)if __name__ __main__:函数(参数)函数(参数2)函数的参数是: 字典 import requests# 定义一个字典 data {} 地址 "https://webdriveruniversity.com/" 请求方法 getdata["url"…

【数据结构】选择题错题集

这里注意原本p后面也是有节点的。 这里只有遍历前面的链表找到尾节点连接即可。 快排是交换排序。 不要想象只有这两个节点,还有其他节点省略了。 筛选法就是向下调整算法。用向下调整建堆从最后一个节点的父亲开始。 这里错位相减法是乘4,所以最后要除三…

超详细!外婆都能看懂的Stable Diffusion入门教程,AI绘画SD零基础入门到精通教程!

一、前言 如今的AI绘画界有两大最强工具,一个是Midjourney ,一个是StableDiffusion。Midjourney生成图片的质量非常高,可问题就是 Midjourney 的图片怎么精准的控制,或者是不改变某个物体的情况下更换背景等等,实在是…

虚拟化数据恢复—断电导致虚拟机目录项被破坏的数据恢复案例

虚拟化数据恢复环境: 某品牌服务器(部署VMware EXSI虚拟机)同品牌存储(存放虚拟机文件)。 虚拟化故障: 意外断电导致服务器上某台虚拟机无法正常启动。查看虚拟机配置文件发现这台故障虚拟机除了磁盘文件以…

视频监控平台是如何运作的?EasyCVR视频汇聚平台的高效策略与实践

随着科技的飞速发展,视频监控平台在社会安全、企业管理、智慧城市构建等领域发挥着越来越重要的作用。一个高效的视频监控平台,不仅依赖于先进的硬件设备,更离不开强大的视频处理技术作为支撑。这些平台集成了多种先进的视频技术,…

长短期记忆网络和UKF的结合|Python代码例程

结合长短期记忆网络(LSTM)和无迹卡尔曼滤波器(UKF)的技术在机器人导航和状态估计中具有广泛的应用前景。 文章目录 结合LSTM和UKF的背景结合LSTM和UKF的优势应用实例研究现状Python代码示例结合LSTM和UKF的背景 长短期记忆网络(LSTM)是一种特殊的递归神经网络(RNN),…

51单片机-IIC实验1-AT24C02数据存储(实战1)

本实验主要是针对IIC的具体案例进行实战,主要利用支持IIC通信的芯片AT24C02进行与51单片机构成通信。51(AT89C52)本身不带有IIC通信,所以,我们需要给51写一个IIC时序,以便与支持IIC协议的AT24C02数据存储芯…

逆向学习系列(三)Charles配合Drony的使用(真机)

在抓包中,有时可能出现无法抓包的情况,这时,可以尝试使用Drony转发抓包。 一、安装Drony 将电脑上的Drony_1.3.154_APKPure.apk复制粘贴进真机的Download文件夹中。在真机上找到这个apk,点击安装。 二、在VMOS中安装Drony 在真…

【Petri网导论学习笔记】Petri网导论入门学习(四)

Petri网导论入门学习(四) Petri 网导论学习笔记(4)1.2 标识网与网系统定义 1.8定义 1.9例 1.4存在空标识网的几种情况1.2 小结1.2学习完应达到的要求 Petri 网导论学习笔记(4) 如需学习转载请注明原作者并附…

55页可编辑PPT | 集团制造企业数字化转型顶层设计方案

这份PPT文档是一份关于集团制造企业数字化转型的顶层业务设计方案。文档详细介绍了企业在后ERP时代面临的挑战,以及如何通过Oracle解决方案来实现数字化转型。 数字化转型的三大要点集中在满足利益相关者的期望,以企业价值为核心引领业务模式的创新&…