Android平台RTMP推送|轻量级RTSP服务|GB28181设备接入模块之实时快照保存JPG还是PNG?

简介: Android平台RTMP推送|轻量级RTSP服务|GB28181设备接入模块之实时快照保存JPG还是PNG?

JPG还是PNG?

JPG和PNG是两种常见的图片文件格式,在压缩方式、图像质量、透明效果和可编辑性等方面存在显著差异。

    • 压缩方式:JPG是一种有损压缩格式,通过丢弃图像数据来减小文件大小,因此可能会损失一些图像细节和质量。而PNG使用的是无损压缩格式,它不会丢失任何原始图像数据,从而保持了图像的完整性和质量。
    • 图像质量:由于压缩方式的不同,JPG在压缩后会牺牲一部分图像数据,因此在图像质量上可能存在损失,例如可能会出现锯齿状边缘或颜色失真。相比之下,PNG的无损压缩可以保证原图像数据的完整性,其256个透明层次的设定可以使图片边缘平滑融合,从而消除图片锯齿边缘。
    • 透明效果:PNG支持透明度,可以用作背景透明的图片,而JPG则不支持透明效果。因此,如果你需要制作半透明的图像或者需要背景透明的图片,PNG是一个更好的选择。
    • 可编辑性:JPG是一种不可编辑的图片格式,一旦被保存为JPG格式,就无法进行修改。而PNG是一种可编辑的图片格式,可以通过图像编辑软件(如Photoshop)进行修改、编辑和重新保存。例如,你可以改变PNG图片中的文字样式、线条等元素。

    Android推流端的截图设计

    大牛直播SDK早期在做Android平台RTMP推流和轻量级RTSP服务模块的时候,截图考虑到PNG的特性,直接保存png图片,随着GB28181-2022规范的实施,规范里面有明确要求,需要支持JPG编码,为此我们针对截图这块,做了如下的调整(对应:实时快照):

    image.gif2023-05-25-3.jpg

    原接口:

    /*** 请使用新的CaptureImage接口, 这个接口只能保存PNG图片, 不推荐使用* Save current image during publishing stream(实时快照)** @param imageName: image name, which including fully path, "/sdcard/daniuliveimage/daniu.png", etc.** @return {0} if successful*/publicnativeintSmartPublisherSaveCurImage(longhandle,  StringimageName);

    image.gif

    值得注意的是,原接口如果需要截图,还需要调用SmartPublisherSaveImageFlag()。

    新的接口,我们设计如下:

    /*** 新的截图接口, 支持JPEG和PNG两种格式* @param compress_format: 压缩格式, 0:JPEG格式, 1:PNG格式, 其他返回错误* @param quality: 取值范围:[0, 100], 值越大图像质量越好, 仅对JPEG格式有效, 若是PNG格式,请填100* @param file_name: 图像文件名, 例如:/dirxxx/test20231113100739.jpeg, /dirxxx/test20231113100739.png* @param user_data_string: 用户自定义字符串* @return {0} if successful*/publicnativeintCaptureImage(longhandle, intcompress_format, intquality, Stringfile_name, Stringuser_data_string);

    image.gif

    如何调用?

    废话不多说,直接上代码:

    privateSimpleDateFormatcapture_image_date_format_;
    classButtonCaptureImageListenerimplementsOnClickListener {
    @SuppressLint("SimpleDateFormat")
    publicvoidonClick(Viewv) {
    if(isPushingRtmp||isRecording||isRTSPPublisherRunning||isPushingRtsp)
                {
    if (null==capture_image_date_format_)
    capture_image_date_format_=newSimpleDateFormat("yyyyMMddHHmmssSSS");
    StringtimeStamp=capture_image_date_format_.format(newDate());
    StringimageFileName=timeStamp;    //创建以时间命名的文件名称StringimagePath=imageSavePath+"/"+imageFileName;
    intquality;
    booleanis_jpeg=true;
    if (is_jpeg) {
    imagePath+=".jpeg";
    quality=100;
                    }
    else {
    imagePath+=".png";
    quality=100;
                    }
    intcapture_ret=libPublisher.CaptureImage(publisherHandle,is_jpeg?0:1, quality, imagePath, "test_user_data");
    Log.i(TAG, "capture_image ret:"+capture_ret+", file:"+imagePath);
                }
    else            {
    Log.e(TAG, "快照失败,请确保在推送、录像或内置RTSP服务发布状态..");
                }
            }
        }

    image.gif

    截图成功,对应的event回调如下:

    classEventHandeV2implementsNTSmartEventCallbackV2 {
    @OverridepublicvoidonNTSmartEventCallbackV2(longhandle, intid, longparam1, longparam2, Stringparam3, Stringparam4, Objectparam5) {
    Log.i(TAG, "EventHandeV2: handle="+handle+" id:"+id);
    Stringpublisher_event="";
    switch (id) {
                    .....
    caseNTSmartEventID.EVENT_DANIULIVE_ERC_PUBLISHER_CAPTURE_IMAGE:
    publisher_event="快照: "+param1+" 路径:"+param3;
    if (0==param1) {
    rename_image_file_name(param3, param2);
    publisher_event=publisher_event+"截取快照成功.."+", 用户数据:"+param4;
                        } elsepublisher_event=publisher_event+"截取快照失败..";
    break;
                        ....
            }
        }

    image.gif

    如果需要对截图后的文件重命名(比如gb28181,我们会把截图时间返上来),便于统一管理,参考代码如下:

    privatevoidrename_image_file_name(Stringfile_name, longfile_date_time_ms) {
    if (null==file_name||file_name.isEmpty()
    ||file_date_time_ms<1||null==capture_image_date_format_)
    return;
    try {
    java.io.Filefile=newFile(file_name);
    if (!file.exists() ||!file.isFile() ||!file.canRead() ||file.length() <1)
    return;
    Stringfile_name_extension=null;
    intindex=file_name.lastIndexOf('.');
    if (index>-1)
    file_name_extension=file_name.substring(index+1);
    Datefile_date=newDate(file_date_time_ms);
    Stringnew_file_name=capture_image_date_format_.format(file_date);
    if (file_name_extension!=null&&!file_name_extension.isEmpty())
    new_file_name+="."+file_name_extension;
    java.io.Filenew_file=newjava.io.File(file.getParent(), new_file_name);
    if (file.renameTo(new_file))
    Log.i(TAG, "rename image file name ok, file_name:"+file_name+", new:"+new_file_name);
    elseLog.e(TAG, "rename image file name failed, file_name:"+file_name);
            } catch (Exceptione) {
    Log.e(TAG, "rename_image_file_name Exception:", e);
            }
        }

    image.gif

    总结

    Android平台RTMP推送、轻量级RTSP还是GB28181设备对接模块,选择哪种图片格式主要取决于具体的使用需求。如果你需要压缩图像文件并且不关心原始图像的完整性,JPG可能是一个更好的选择。而如果你需要保持原始图像的完整性和质量,或者需要制作背景透明的图片,那么PNG可能是更好的选择。

    相关文章
    |
    1月前
    |
    IDE 开发工具 Android开发
    移动应用开发之旅:探索Android和iOS平台
    在这篇文章中,我们将深入探讨移动应用开发的两个主要平台——Android和iOS。我们将了解它们的操作系统、开发环境和工具,并通过代码示例展示如何在这两个平台上创建一个简单的“Hello World”应用。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和技巧,帮助你更好地理解和掌握移动应用开发。
    68 17
    |
    3月前
    |
    Linux API 开发工具
    FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
    ijkplayer是由B站研发的移动端播放器,基于FFmpeg 3.4,支持Android和iOS。其源码托管于GitHub,截至2024年9月15日,获得了3.24万星标和0.81万分支,尽管已停止更新6年。本文档介绍了如何在Linux环境下编译ijkplayer的so库,以便在较新的开发环境中使用。首先需安装编译工具并调整/tmp分区大小,接着下载并安装Android SDK和NDK,最后下载ijkplayer源码并编译。详细步骤包括环境准备、工具安装及库编译等。更多FFmpeg开发知识可参考相关书籍。
    122 0
    FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
    |
    开发工具 Android开发
    Android 7.1 使用mmm编译模块失败
    Android 7.1 使用mmm编译模块失败
    311 0
    |
    Android开发
    Android不编译某个模块
    Android 5.1 源码,编译相关的文件一般在build目录下build/target/product 放了很多mk文件;一般不同的产品会有不同的目录 假设我不想编译OpenWnn,在build目录下grep一下“OpenWnn”target/product/full_base.
    1445 0
    |
    2月前
    |
    开发框架 前端开发 Android开发
    安卓与iOS开发中的跨平台策略
    在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
    |
    2月前
    |
    缓存 前端开发 Android开发
    安卓开发中的自定义视图:从零到英雄
    【10月更文挑战第42天】 在安卓的世界里,自定义视图是一块画布,让开发者能够绘制出独一无二的界面体验。本文将带你走进自定义视图的大门,通过深入浅出的方式,让你从零基础到能够独立设计并实现复杂的自定义组件。我们将探索自定义视图的核心概念、实现步骤,以及如何优化你的视图以提高性能和兼容性。准备好了吗?让我们开始这段创造性的旅程吧!
    37 1
    |
    1月前
    |
    搜索推荐 前端开发 API
    探索安卓开发中的自定义视图:打造个性化用户界面
    在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
    57 19
    |
    2月前
    |
    IDE Java 开发工具
    移动应用与系统:探索Android开发之旅
    在这篇文章中,我们将深入探讨Android开发的各个方面,从基础知识到高级技术。我们将通过代码示例和案例分析,帮助读者更好地理解和掌握Android开发。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和技巧。让我们一起开启Android开发的旅程吧!
    |
    1月前
    |
    JSON Java API
    探索安卓开发:打造你的首个天气应用
    在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
    64 14
    |
    1月前
    |
    Java Linux 数据库
    探索安卓开发:打造你的第一款应用
    在数字时代的浪潮中,每个人都有机会成为创意的实现者。本文将带你走进安卓开发的奇妙世界,通过浅显易懂的语言和实际代码示例,引导你从零开始构建自己的第一款安卓应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇门,让你的创意和技术一起飞扬。

    热门文章

    最新文章