基于 FFmpeg 的跨平台视频播放器简明教程(十二):Android SurfaceView 显示图片和播放视频

系列文章目录

  1. 基于 FFmpeg 的跨平台视频播放器简明教程(一):FFMPEG + Conan 环境集成
  2. 基于 FFmpeg 的跨平台视频播放器简明教程(二):基础知识和解封装(demux)
  3. 基于 FFmpeg 的跨平台视频播放器简明教程(三):视频解码
  4. 基于 FFmpeg 的跨平台视频播放器简明教程(四):像素格式与格式转换
  5. 基于 FFmpeg 的跨平台视频播放器简明教程(五):使用 SDL 播放视频
  6. 基于 FFmpeg 的跨平台视频播放器简明教程(六):使用 SDL 播放音频和视频
  7. 基于 FFmpeg 的跨平台视频播放器简明教程(七):使用多线程解码视频和音频
  8. 基于 FFmpeg 的跨平台视频播放器简明教程(八):音画同步
  9. 基于 FFmpeg 的跨平台视频播放器简明教程(九):Seek 策略
  10. 基于 FFmpeg 的跨平台视频播放器简明教程(十):在 Android 运行 FFmpeg
  11. 基于 FFmpeg 的跨平台视频播放器简明教程(十一):一种简易播放器的架构介绍

前言

上一章中我们介绍了一个简易的播放器架构,对之前零碎的代码片段进行了组织和重构,形成了较为灵活的一种架构设计,它非常简单,但足够满足我们的需求。

现在,接着我们在 Android 上的旅程。今天我们来讨论如何在 Android 上显示画面。

Android 原生的 Java/Kotlin 接口播放视频还是很容易的,有 MediaController、MediaPlayer 等类可以直接使用,相关教程参考Android实现视频播放的3种实现方式。

由于我们的代码几乎都是 C/C++ ,因此需要找到一种从 Native 层进行视频播放的方法。这里要介绍的是 SurfaceView + Native 的显示方式。

Android 图像绘制介绍

关于 Android 图像绘制系统网上有很多文章说的较为清楚了,例如

  • 从整体上看Android图像显示系统
  • Android图形系统综述(干货篇)

这些内容中,我们要重点关注 BufferQueue 中生产者与消费者之间的关系
在这里插入图片描述

例如:一个Activity是一个Surface、一个Dialog也是一个Surface,承载了上层的图形数据,与SurfaceFlinger侧的Layer相对应。

Native层Surface实现了ANativeWindow结构体,在构造函数中持有一个IGraphicBufferProducer,用于和 BufferQueue 进行交互。
————————————————
版权声明:本文为CSDN博主「Jason_Lee155」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/jason_lee155/article/details/121663662

有了上述的基本认识后,接下来我们介绍 SurfaceView

Android 中 Surface 与 SurfaceView

关于 Surface 的介绍文章有:

  • 浅谈Android Surface机制
  • 深入Android系统(十二)Android图形显示系统-1-显示原理与Surface

总结下上述文章的对我们来说重要的理解内容:

  1. 无论开发者使用什么渲染API,一切内容都会渲染到Surface上;Surface中会关联一个BufferQueue用于提供图像数据缓存
  2. Suface 继承自 ANativeWindow,可以通过 dequeueBuffer,queueBuffer,lockBuffer 等接口拿到 BufferQueue 中的 Buffer 对象

关于 SurfaceView 的介绍文章有:

  • Android音视频开发系列-SurfaceView介绍
  • SurfaceView 与 TextureView 详解

总结下上述文章对我们来说的重点内容:

  1. SurfaceView和宿主窗口是分离的。正常情况下窗口的View共享同一个Window,而Window也对应一个Surface,所有View也就共享同一个Surface。而 SurfaceView 具备独立的Surface,相当于和宿主窗口绘制是分离互不干扰。
  2. SurfaceView 的核心在于提供了两个线程:UI线程和渲染线程,两个线程通过“双缓冲”机制来达到高效的界面适时更新

OK,我们将上面的知识串起来:

  1. SurfaceView 持有一个 Surface。
  2. 通过 Surface 我们能够获取到 BufferQueue 中的一个 Buffer。
  3. 将图像绘制到这个 Buffer 后,就能正确的显示图像。
  4. SurfaceView 中的 Surface 与其他窗口的 View 是独立的,我们可以在另一个线程中去渲染它

JNI SurfaceView 显示图片

在了解了 Surface 和 SurfaceView 后,我们现在已经能够做到使用 JNI(NDK)和 SurfaceView 来显示一张图片了,具体代码在 T02DisplayImageActivity 中,现在做一些代码上的解释。

class T02DisplayImageActivity : AppCompatActivity() {private lateinit var mSurfaceView: MySurfaceViewoverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_display_image)// get image bitmapval options = BitmapFactory.Options()options.inScaled = falseval bitmap = BitmapFactory.decodeResource(resources, R.drawable.test, options)mSurfaceView = findViewById(R.id.surfaceView_display_image)mSurfaceView.setAspectRation(bitmap.width, bitmap.height)mSurfaceView.holder.addCallback(object: SurfaceHolder.Callback{override fun surfaceCreated(holder: SurfaceHolder) {val surface = holder.surfacerenderImage(surface, bitmap)}override fun surfaceChanged(p0: SurfaceHolder, format: Int, width: Int, height: Int) {}override fun surfaceDestroyed(p0: SurfaceHolder) {}})}external fun renderImage(surface: Surface, bitmap: Bitmap);
}
  1. 我们在 R.layout.activity_display_image 中布局中放置一个 MySurfaceView 用于显示图片。MySurfaceView 继承自 SurfaceView,并且做了一些方法的重写,这部分后面再细说。
  2. 通过 BitmapFactory 来读取一张图片,用于显示
  3. 接下来,添加了一个SurfaceHolder的回调函数,用于监听SurfaceView的生命周期。具体来说:surfaceCreated()方法在SurfaceView创建时被调用,其中的holder参数表示SurfaceView的SurfaceHolder对象,可以通过它获取到Surface对象。在这个方法中,调用了renderImage()函数,将bitmap渲染到Surface上。
  4. 现在看看 renderImage 方法做了什么,代码如下:

extern "C"
JNIEXPORT void JNICALL
Java_com_example_videoplayertutorials_T02DisplayImageActivity_renderImage(JNIEnv *env,jobject thiz,jobject surface,jobject bitmap) {AndroidBitmapInfo info;AndroidBitmap_getInfo(env, bitmap, &info);char *data = NULL;AndroidBitmap_lockPixels(env, bitmap, (void **) &data);ANativeWindow *nativeWindow = ANativeWindow_fromSurface(env, surface);ANativeWindow_setBuffersGeometry(nativeWindow, info.width, info.height,AHARDWAREBUFFER_FORMAT_R8G8B8A8_UNORM);ANativeWindow_Buffer buffer;ANativeWindow_lock(nativeWindow, &buffer, NULL);auto *data_src_line = (int32_t *) data;const auto src_line_stride = info.stride / sizeof(int32_t);auto *data_dst_line = (uint32_t *) buffer.bits;for (int y = 0; y < buffer.height; y++) {std::copy_n(data_src_line, buffer.width, data_dst_line);data_src_line += src_line_stride;data_dst_line += buffer.stride;}ANativeWindow_unlockAndPost(nativeWindow);AndroidBitmap_unlockPixels(env, bitmap);ANativeWindow_release(nativeWindow);
}
  1. 通过AndroidBitmap_getInfo函数获取Bitmap的信息,包括宽度、高度、格式等。

  2. 通过AndroidBitmap_lockPixels函数锁定Bitmap的像素,防止在操作过程中被其他线程修改。

  3. 通过ANativeWindow_fromSurface函数获取Surface对应的ANativeWindow。

  4. 通过ANativeWindow_setBuffersGeometry函数设置ANativeWindow的缓冲区大小和格式。注意,这里我们设置的格式是 AHARDWAREBUFFER_FORMAT_R8G8B8A8_UNORM,也就是 RGBA 格式,因为我们的数据源 bitmap 它就是 RGBA 格式的。你可以在 AHardwareBuffer_Format 找到所有支持的格式,其中 YUV420 也是支持的。

  5. 通过ANativeWindow_lock函数锁定ANativeWindow的缓冲区,准备写入数据。

  6. 将Bitmap的像素数据复制到ANativeWindow的缓冲区。

  7. 通过ANativeWindow_unlockAndPost函数解锁ANativeWindow的缓冲区,并将缓冲区的内容显示到屏幕上。

  8. 通过AndroidBitmap_unlockPixels函数解锁Bitmap的像素。

  9. 通过ANativeWindow_release函数释放ANativeWindow。

如果你的代码正确,那么可以看到图片正常显示,如下图:

正确的 View 大小

为了说明 View 大小的问题,让我们先将代码中的 MySurfaceView 全部改为 SurfaceView,运行代码后,你会发现图片显示时被拉伸填充了,如下图:

额,所以这是为啥嘞?让我们来分析分析:

  1. 首先,在 ANativeWindow_setBuffersGeometry 中,我们对 Buffer 大小和格式进行了设置,它的大小与图片大小是一致的。只有与 Buffer 大小一致,才能正确地将所有图片数据拷贝到 Buffer 上。所以这里是没问题的。
  2. 通过 Android Studio 的 Layout Inspector 工具,可以看到 Surface View 填满了整个屏幕
    在这里插入图片描述
  3. 也就是说 Buffer 中图片,被 Android 系统做了 resize 使其填充到整个屏幕了。因此导致了图像的拉伸情况。

了解了原因,那么如何解决这个问题呢?大致思路是去修改 view 的尺寸,让 view 适配视频的尺寸。具体的:

  1. 新建 MySurfaceView,继承自 SurfaceView
  2. 在 MySurfaceView 新增 setAspectRation 方法,设置 view 的宽高比,例如 16:9 是一个横屏的宽高比
  3. 接着,重写 onMeasure 方法,在 onMeasure 方法中根据宽高比, 具体代码如下:
override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {super.onMeasure(widthMeasureSpec, heightMeasureSpec)val width = MeasureSpec.getSize(widthMeasureSpec)val height = MeasureSpec.getSize(heightMeasureSpec)if(mWidth == 0){setMeasuredDimension(width, height)return}// calculate expected width by ratioval expectedWidth = height * mWidth / mHeight// if expected width is too big, set max width to expected widthif(expectedWidth >= width){// to maintain aspect ratio, calculate expected heightval expectedHeight = width * mHeight / mWidthsetMeasuredDimension(width, expectedHeight)}else{// or the expected width can fit in the parent, set the expected widthsetMeasuredDimension(expectedWidth, height)}}
  • 首先,调用父类方法 super.onMeasure 和 MeasureSpec.getSize 获取预期的 width 和 height。例如 width=1080, height=2070,即屏幕的大小
  • 接着,根据预期的宽高比去计算 expectedWidth,例如宽高比是 16:9 ,那么 e x p W h = 16 9 \frac{expW}{h}=\frac{16}{9} hexpW=916 ,得到 expectedWidth = height * 16/9 = 2070 * 16/9 = 3680
  • 如果 expectedWidth >= width 说明如果按照宽高比对现有 view 进行等比例放大,那么超过目前可接受的最大宽度,无法满足。因此,我们转而去缩放 height,以便最终 view 的宽高比符合我们的预期。因此 w e x p H = 16 9 \frac{w}{expH}=\frac{16}{9} expHw=916 得到 val expectedHeight = width * mHeight / mWidth
  • 如果 expectedWidth < width 说明当前的 view 可以放下,则直接设置 setMeasuredDimension(expectedWidth, height) 即可

完成了上述的修改,我们使用 MySurfaceView 进行视频的显示,此时 View 的尺寸符合图片的宽高比,图片也不会被拉伸和缩放了
在这里插入图片描述

SurfaceView 播放视频

我们了解了如何使用 SurfaceView 显示图片,并解决了图片被拉伸的问题。显示视频那也就水到渠成了,视频只是很多张图片罢了。
首先,我们将上屏显示图像的模块进行封装,在 基于 FFmpeg 的跨平台视频播放器简明教程(十一):一种简易播放器的架构介绍 文中,VideoOutput 模块负责显示视频帧,我们新建一个叫 SurfaceViewVideoOutput 的类,用它来在 SurfaceView 上显示图片。具体代码在 j_andr_surfaceview_video_output 中,其中 drawFrame 完全与显示图片的代码是一致的:

int drawFrame(std::shared_ptr<Frame> frame) override {if(nativeWindow_ == nullptr) {LOGE("nativeWindow_ is null, can't drawFrame");return -1;}ANativeWindow_Buffer buffer;ANativeWindow_lock(nativeWindow_, &buffer, NULL);// copy frame to bufferauto *data_src_line = (int32_t *) frame->f->data[0];const auto src_line_stride = frame->f->linesize[0] / sizeof(int32_t);auto *data_dst_line = (uint32_t *) buffer.bits;auto height = std::min(buffer.height, frame->f->height);for (int y = 0; y < height; y++) {std::copy_n(data_src_line, buffer.width, data_dst_line);data_src_line += src_line_stride;data_dst_line += buffer.stride;}ANativeWindow_unlockAndPost(nativeWindow_);return 0;}

接着,为了在 Android 上更容易使用 C/C++ 播放器代码,我们创建一个 SimplePlayer 的 Kotlin 类,它是对底层 j_video_player::SimplePlayer 的封装,所有与播放相关接口都通过 SimplePlayer 最终调用到 C/C++ 层。具体代码查看 SimplePlayer 以及它的 JNI 实现 jni_simple_player

完成了上面的动作,我们在 Android 上就能愉快地进行视频播放了,具体代码在 DisplayVideoActivity 中。

总结

本文首先简略的介绍了 Android 图像的显示系统,引出 BufferQueue 的概念;接着介绍了 Surface 和 SurfaceView,Surface 关联着一个 BufferQueue,而 SurfaceView 持有一个 Surface;接下来,我们展示了如何在 SurfaceView 上显示图片,并解决图片宽高比与手机屏幕不一致导致的图像拉伸问题;最后,我们使用 SimplePlayer 在 SurfaceView 做视频播放。

参考

  • Android实现视频播放的3种实现方式
  • 从整体上看Android图像显示系统
  • Android图形系统综述(干货篇)

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

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

相关文章

【LeetCode:2866. 美丽塔 II | 单调栈 + 前后缀数组】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

flink watermark 实例分析

WATERMARK 定义了表的事件时间属性&#xff0c;其形式为: WATERMARK FOR rowtime_column_name AS watermark_strategy_expression rowtime_column_name 把一个现有的列定义为一个为表标记事件时间的属性。该列的类型必须为 TIMESTAMP(3)/TIMESTAMP_LTZ(3)&#xff0c;且是 sche…

2023年12月GESP认证图形化编程四级真题试卷

2023年12月GESP认证Scratch图形化等级考试&#xff08;四级&#xff09;真题试卷 题目总数&#xff1a;27 总分数&#xff1a;100 选择题 第 1 题 单选题 现代计算机是指电子计算机&#xff0c;它所基于的是&#xff08; &#xff09;体系结构 A. 艾伦图灵 B. …

Valentina Studio Pro for Mac:高效数据库管理工具

作为一款强大而高效的数据库管理工具&#xff0c;Valentina Studio Pro for Mac在Mac平台上的表现无疑是令人印象深刻的。无论您是初学者还是专业数据库管理员&#xff0c;Valentina Studio Pro都能够满足您的需要&#xff0c;并提供一流的工具和功能来简化数据库管理的过程。 …

KBU808-ASEMI适配高端电源KBU808

编辑&#xff1a;ll KBU808-ASEMI适配高端电源KBU808 型号&#xff1a;KBU808 品牌&#xff1a;ASEMI 封装&#xff1a;KBU-4 最大平均正向电流&#xff1a;8A 最大重复峰值反向电压&#xff1a;800V 产品引线数量&#xff1a;4 产品内部芯片个数&#xff1a;4 产品内…

Docker 编译OpenHarmony 4.0 release

一、背景介绍 1.1、环境配置 编译环境&#xff1a;Ubuntu 20.04OpenHarmony版本&#xff1a;4.0 release平台设备&#xff1a;RK3568 OpenHarmony 3.2更新至OpenHarmony 4.0后&#xff0c;公司服务器无法编译通过&#xff0c;总是在最后几十个文件时报错,错误码4000&#xf…

C#电源串口调试

目的 记录串口调试的遇到的一些问题以及相应的解决方法 1.串口定义:串口是计算机与其他硬件传输数据的通道&#xff0c;在计算机与外设通信时起到重要作用 2.串口通信的基础知识 C#中的串口通信类 C#使用串口通信类是SerialPort(),该类使用方法是 new 一个 SerialPort对象 为S…

【C语言】自定义类型:结构体深入解析(二)结构体内存对齐宏offsetof计算偏移量结构体传参

文章目录 &#x1f4dd;前言&#x1f320; 结构体内存对齐&#x1f309;内存对齐包含结构体的计算&#x1f320;宏offsetof计算偏移量&#x1f309;为什么存在内存对⻬?&#x1f320; 结构体传参&#x1f6a9;总结 &#x1f4dd;前言 本小节&#xff0c;我们学习结构的内存对…

R语言【cli】——通过cli_abort用 cli 格式的内容显示错误、警告或信息,内部调用cli_bullets和inline-makeup

cli_abort(message,...,call .envir,.envir parent.frame(),.frame .envir ) 先从那些不需要下大力气理解的参数入手&#xff1a; 参数【.envir】&#xff1a;进行万能表达式编译的环境。 参数【.frame】&#xff1a;抛出上下文。默认用于参数【.trace_bottom】&#xff…

Redis实现日榜|直播间榜单|排行榜|Redis实现日榜01

前言 直播间贡献榜是一种常见的直播平台功能&#xff0c;用于展示观众在直播过程中的贡献情况。它可以根据观众的互动行为和贡献值进行排名&#xff0c;并实时更新&#xff0c;以鼓励观众积极参与直播活动。 在直播间贡献榜中&#xff0c;每个观众都有一个对应的贡献值&#…

力扣日记12.21【二叉树篇】98. 验证二叉搜索树

力扣日记&#xff1a;【二叉树篇】98. 验证二叉搜索树 日期&#xff1a;2023.12.21 参考&#xff1a;代码随想录、力扣 98. 验证二叉搜索树 题目描述 难度&#xff1a;中等 给你一个二叉树的根节点 root &#xff0c;判断其是否是一个有效的二叉搜索树。 有效 二叉搜索树定义…

啥?你还不道数据库?赶紧进来看吧!

操作系统&#xff1a; windows&#xff1a;win10、win11、win7、windows Server2016 Linux/Unix &#xff1a;红帽&#xff08;RedHat&#xff09;、Bebian、SUSE MacOS Linux系统&#xff1a;CantOS&#xff08;yum、dnf&#xff09;、Ubuntu&#xff08;apt、apt—get&am…

Ubuntu 常用命令之 df 命令用法介绍

&#x1f4d1;Linux/Ubuntu 常用命令归类整理 在Ubuntu系统下&#xff0c;df命令是用来查看文件系统的磁盘空间占用情况的。df是disk free的缩写&#xff0c;这个命令可以获取硬盘被占用了多少空间&#xff0c;还有多少空间是可用的&#xff0c;硬盘的挂载点等信息。 df命令的…

【Python】matplotlib画图_饼状图

柱状图主要使用pie()函数&#xff0c;基本格式如下&#xff1a; plt.pie(x,explodeNone,labelsNone,colorsNone,autopctsNone,pctdistance0.6,shadowFalse,labeldistance1.1,staatangleNone,radiusNone,counterclockTrue,wedgepropsNone,textpropsNone,center(0,0),frameFalse…

PIC单片机项目(7)——基于PIC16F877A的智能灯光设计

1.功能设计 使用PIC16F877A单片机&#xff0c;检测环境关照&#xff0c;当光照比阈值低的时候&#xff0c;开灯。光照阈值可以通过按键进行设置&#xff0c;同时阈值可以保存在EEPROM中&#xff0c;断电不丢失。使用LCD1602进行显示&#xff0c;第一行显示测到的实时光照强度&a…

代码随想录算法训练营Day7 | 344.反转字符串、541.反转字符串||、替换数字、151.反转字符串中的单词、右旋字符串

LeetCode 344 反转字符串 本题思路&#xff1a;反转字符串比较简单&#xff0c;定义两个指针&#xff0c;一个 i 0, 一个 j s.length-1。然后定义一个临时变量 tmp&#xff0c;进行交换 s[i] 和 s[j]。 class Solution {public void reverseString(char[] s) {int i 0;int …

华为二层交换机与防火墙配置实例

二层交换机与防火墙对接上网配置示例 组网图形 图1 二层交换机与防火墙对接上网组网图 二层交换机简介配置注意事项组网需求配置思路操作步骤配置文件相关信息 二层交换机简介 二层交换机指的是仅能够进行二层转发&#xff0c;不能进行三层转发的交换机。也就是说仅支持二层…

OceanMind海睿思入选中国信通院首批“高质量智能审计工具目录”,获多项认证

近日&#xff0c;由中国信息通信研究院&#xff08;以下简称“中国信通院”&#xff09;、中国通信标准化协会支持的“2023 GOLF IT新治理领导力论坛”在北京顺利举行。 中新赛克海睿思作为国内领先的审计数字化代表企业受邀参会。 在内部审计数字化转型走深向实以及智能化演进…

【Spring Boot】面试题汇总,带答案的那种

继上次的文章【MySQL连环炮&#xff0c;你抗的住嘛&#xff1f;】爆火之后&#xff0c;越来越多的小伙伴后台留言&#xff0c;要求阿Q总结下其他的“连环炮”知识点&#xff0c;想在金九银十的面试黄金期轻松对线面试官。 同样为了节省大家的时间&#xff0c;阿Q最近对【Sprin…

性能优化之资源优化

性能优化之资源优化 资源优化性能关键检测流程。浅析一下基于Unity3D 美术规则约束一、模型层面二、贴图层面三、动画层面四、声音层面&#xff1a;&#xff08;音频通用设置&#xff09;五、UI层面&#xff1a; 题外点&#xff1a;诚然在优化中&#xff0c;美术占比是很重要的…