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可能是更好的选择。

    相关文章
    |
    17天前
    |
    编解码 开发工具 Android开发
    Android平台RTMP直播推送模块技术接入说明
    大牛直播SDK跨平台RTMP直播推送模块,始于2015年,支持Windows、Linux(x64_64架构|aarch64)、Android、iOS平台,支持采集推送摄像头、屏幕、麦克风、扬声器、编码前、编码后数据对接,功能强大,性能优异,配合大牛直播SDK的SmartPlayer播放器,轻松实现毫秒级的延迟体验,满足大多数行业的使用场景。RTMP直播推送模块数据源,支持编码前、编码后数据对接
    |
    17天前
    |
    开发工具 Android开发 开发者
    Android平台如何不推RTMP|不发布RTSP流|不实时录像|不回传GB28181数据时实时快照?
    本文介绍了一种在Android平台上实现实时截图快照的方法,尤其适用于无需依赖系统接口的情况,如在RTMP推送、RTSP服务或GB28181设备接入等场景下进行截图。通过底层模块(libSmartPublisher.so)实现了截图功能,封装了`SnapShotImpl.java`类来管理截图流程。此外,提供了关键代码片段展示初始化SDK实例、执行截图、以及在Activity销毁时释放资源的过程。此方案还考虑到了快照数据的灵活处理需求,符合GB/T28181-2022的技术规范。对于寻求更灵活快照机制的开发者来说,这是一个值得参考的设计思路。
    |
    16天前
    |
    图形学 Android开发 iOS开发
    穿越数字洪流,揭秘Unity3d中的视频魔法!Windows、Android和iOS如何征服RTSP与RTMP的终极指南!
    【8月更文挑战第15天】在数字媒体的海洋中,实时视频流是连接世界的桥梁。对于那些渴望在Unity3d中搭建这座桥梁的开发者来说,本文将揭示如何在Windows、Android和iOS平台上征服RTSP与RTMP的秘密。我们将深入探讨这两种协议的特性,以及在不同平台上实现流畅播放的技巧。无论你是追求稳定性的RTSP拥趸,还是低延迟的RTMP忠实粉丝,这里都有你需要的答案。让我们一起穿越数字洪流,探索Unity3d中视频魔法的世界吧!
    33 2
    |
    1天前
    |
    监控 Java 开发工具
    如何快速对接Android平台GB28181接入模块(SmartGBD)
    大牛直播SDK推出的Android平台GB28181接入SDK(SmartGBD),可实现不具备国标音视频能力的 Android终端,通过平台注册接入到现有的GB/T28181—2016服务,可用于如执法记录仪、智能安全帽、智能监控、智慧零售、智慧教育、远程办公、明厨亮灶、智慧交通、智慧工地、雪亮工程、平安乡村、生产运输、车载终端等场景,可能是业内为数不多功能齐全性能优异的商业级水准GB28181接入SDK。
    |
    17天前
    |
    监控 开发工具 Android开发
    结合GB/T28181规范探讨Android平台设备接入模块心跳实现
    本文介绍了GB28181标准中的状态信息报送机制,即心跳机制,用于监控设备与服务器间的连接状态。根据国标GB/T28181-2016,设备在异常时需立即发送状态信息,在正常状态下则按固定间隔(默认60秒)定期发送。若连续三次(默认值)未收到心跳,则视为离线。文章展示了在Android平台的GB28181设备接入模块(SmartGBD)中,如何调整心跳间隔为20秒及超时次数为3次,并给出了心跳消息的示例和异常处理代码片段。对于希望深入了解或遇到问题的开发者,作者提供了进一步交流的机会。
    |
    2天前
    |
    测试技术 Linux Android开发
    探索安卓开发之旅:从初学者到专家
    【8月更文挑战第29天】本文是一篇为初学者和有一定经验的开发者准备的安卓开发指南。我们将从基础概念开始,逐步深入到高级主题,如自定义视图、性能优化等。无论你是刚刚入门,还是希望提升自己的技能,这篇文章都将为你提供有价值的信息和建议。让我们一起踏上这段激动人心的旅程吧!
    |
    1天前
    |
    供应链 物联网 区块链
    未来触手可及:探索新兴技术的趋势与应用安卓开发中的自定义视图:从基础到进阶
    【8月更文挑战第30天】随着科技的飞速发展,新兴技术如区块链、物联网和虚拟现实正在重塑我们的世界。本文将深入探讨这些技术的发展趋势和应用场景,带你领略未来的可能性。
    |
    2天前
    |
    存储 搜索推荐 Java
    探索安卓开发中的自定义视图:打造个性化UI组件Java中的异常处理:从基础到高级
    【8月更文挑战第29天】在安卓应用的海洋中,一个独特的用户界面(UI)能让应用脱颖而出。自定义视图是实现这一目标的强大工具。本文将通过一个简单的自定义计数器视图示例,展示如何从零开始创建一个具有独特风格和功能的安卓UI组件,并讨论在此过程中涉及的设计原则、性能优化和兼容性问题。准备好让你的应用与众不同了吗?让我们开始吧!
    |
    1天前
    |
    XML 搜索推荐 Android开发
    安卓开发中的自定义View组件实践
    【8月更文挑战第30天】探索Android世界,自定义View是提升应用界面的关键。本文以简洁的语言带你了解如何创建自定义View,从基础到高级技巧,一步步打造个性化的UI组件。
    |
    2天前
    |
    设计模式 Java Android开发
    探索安卓应用开发:从新手到专家的旅程探索iOS开发中的SwiftUI框架
    【8月更文挑战第29天】本文旨在通过一个易于理解的旅程比喻,带领读者深入探讨安卓应用开发的各个方面。我们将从基础概念入手,逐步过渡到高级技术,最后讨论如何维护和推广你的应用。无论你是编程新手还是有经验的开发者,这篇文章都将为你提供有价值的见解和实用的代码示例。让我们一起开始这段激动人心的旅程吧!
    下一篇
    云函数