关于libyuv格式转换、图像裁剪说明

2023-12-14 18:10

本文主要是介绍关于libyuv格式转换、图像裁剪说明,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

项目源代码https://github.com/liluojun/PlayVideo

关联文章https://blog.csdn.net/hjt505694246/article/details/105046233Android硬编、硬解h264

https://blog.csdn.net/hjt505694246/article/details/105047729FFmpeg解码裸H264数据

         Android相机输出的是yuv数据、yuv数据格式太多了如420p、420sp等,我们经常会需要对yuv数据做处理,最原始的使用java for循环的方式耗时太多不合算,libyuv是由google开源的一个针对yuv数据处理的库,其效率比使用java处理的方式快太多了,故比较推荐使用。

public class NativeMethod {public native void nv21ToI420(byte[] src, byte[] dst, int w, int h, byte[] y, byte[] u, byte[] v);public native void nv21ToNv12(byte[] src, byte[] dst, int w, int h, byte[] y, byte[] u, byte[] v);public native void nv21CutterToI420(byte[] src, byte[] dst, int cutter_w, int cutter_h, int w, int h, byte[] y, byte[] u, byte[] v);public native void nv21CutterToNv12(byte[] src, byte[] dst, int cutter_w, int cutter_h, int w, int h, byte[] y, byte[] u, byte[] v);
}

我在项目中提供了4个方法示例前两个是nv21转420p和420sp,后两个是转的过程中做出裁剪。

参数说明:byte[] src 源数据

                  byte[] dst转换后的输出数据

                  int cutter_w裁剪的宽度

                  int cutter_h裁剪的高度

                  int w原宽度

                  int h原高度

                 byte[] y, byte[] u, byte[] v裁剪后yuv三个分量的数据,由于我用的是yuv渲染故需要这些值,没需求的可以不用理会。

jni层代码

#define  LOGE(...) __android_log_print(ANDROID_LOG_ERROR,"ffmpeg",__VA_ARGS__)
#ifndef _Included_media_jni_NativeMethod
#define _Included_media_jni_NativeMethod
#ifdef __cplusplus
extern "C" {
#include "libyuv.h"
#include <android/log.h>
#include "media_jni_NativeMethod.h"
#endif
void releaseByteArray(JNIEnv *env, jbyteArray array, uint8_t *elems, jint mode) {env->ReleaseByteArrayElements(array, (jbyte *) elems, mode);env->DeleteLocalRef(array);
}
/** Class:     media_jni_NativeMethod* Method:    nv21ToI420* Signature: ([B[BII)V*/
JNIEXPORT void JNICALL Java_media_jni_NativeMethod_nv21ToI420(JNIEnv *env, jobject job, jbyteArray src, jbyteArray dst, jint w, jint h, jbyteArray y,jbyteArray u, jbyteArray v) {uint8_t *srcArray = (uint8_t *) env->GetByteArrayElements(src, NULL);uint8_t *dstArray = (uint8_t *) env->GetByteArrayElements(dst, NULL);uint8_t *yArray = (uint8_t *) env->GetByteArrayElements(y, NULL);uint8_t *uArray = (uint8_t *) env->GetByteArrayElements(u, NULL);uint8_t *vArray = (uint8_t *) env->GetByteArrayElements(v, NULL);jint uvW = w >> 1;jint ySize = w * h;jint uSize = (uvW) * (h >> 1);uint8_t *y_src = srcArray;uint8_t *uv_src = srcArray + ySize;uint8_t *y_dst = dstArray;uint8_t *u_dst = dstArray + ySize;uint8_t *v_dst = dstArray + ySize + uSize;libyuv::NV21ToI420(y_src, w, uv_src, w, y_dst, w, u_dst, uvW, v_dst, uvW, w, h);memcpy(yArray, y_dst, ySize);memcpy(uArray, u_dst, (w * h) >> 2);memcpy(vArray, v_dst, (w * h) >> 2);//I420旋转方法注释但不能删除
//    yuv.libyuv::I420Rotate(y_tran, w, u_tran, w >> 1, v_tran, w >> 1, y_dst, h, u_dst, h >> 1, v_dst,
//                       h >> 1, h, w, yuv.libyuv::kRotate270);releaseByteArray(env, src, srcArray, 0);releaseByteArray(env, dst, dstArray, 0);releaseByteArray(env, y, yArray, 0);releaseByteArray(env, v, vArray, 0);releaseByteArray(env, u, uArray, 0);
} ;/** Class:     media_jni_NativeMethod* Method:    nv21ToNv12* Signature: ([B[BII)V*/
JNIEXPORT void JNICALL Java_media_jni_NativeMethod_nv21ToNv12(JNIEnv *env, jobject job, jbyteArray src, jbyteArray dst, jint w, jint h, jbyteArray y,jbyteArray u, jbyteArray v) {LOGE("111");uint8_t *srcArray = (uint8_t *) env->GetByteArrayElements(src, 0);uint8_t *dstArray = (uint8_t *) env->GetByteArrayElements(dst, 0);LOGE("222");uint8_t *yArray = (uint8_t *) env->GetByteArrayElements(y, NULL);uint8_t *uArray = (uint8_t *) env->GetByteArrayElements(u, NULL);uint8_t *vArray = (uint8_t *) env->GetByteArrayElements(v, NULL);jint ySize = w * h;jint uvSize = (w >> 1) * (h >> 1);uint8_t *y_src = srcArray;uint8_t *uv_src = srcArray + ySize;uint8_t *u_src = srcArray + ySize;uint8_t *v_src = srcArray + ySize + uvSize;uint8_t *y_dst = dstArray;uint8_t *uv_dst = dstArray + ySize;libyuv::NV21ToI420(y_src, w, uv_src, w, yArray, w, uArray, w >> 1, vArray, w >> 1, w,h);libyuv::I420ToNV12(yArray, w, uArray, w >> 1, vArray, w >> 1, y_dst, w, uv_dst, w, w, h);releaseByteArray(env, src, srcArray, 0);releaseByteArray(env, dst, dstArray, 0);releaseByteArray(env, y, yArray, 0);releaseByteArray(env, v, vArray, 0);releaseByteArray(env, u, uArray, 0);
} ;
/** Class:     media_jni_NativeMethod* Method:    nv21CutterToI420* Signature: ([B[BIIII)V*/
JNIEXPORT void JNICALL Java_media_jni_NativeMethod_nv21CutterToI420(JNIEnv *env, jobject job, jbyteArray src, jbyteArray dst, jint cw, jint ch, jint w,jint h, jbyteArray y,jbyteArray u, jbyteArray v) {uint8_t *srcArray = (uint8_t *) env->GetByteArrayElements(src, 0);uint8_t *dstArray = (uint8_t *) env->GetByteArrayElements(dst, 0);jbyteArray transitJbyteArray = env->NewByteArray(w * h * 3 / 2);uint8_t *transitArray = (uint8_t *) env->GetByteArrayElements(transitJbyteArray, 0);uint8_t *yArray = (uint8_t *) env->GetByteArrayElements(y, NULL);uint8_t *uArray = (uint8_t *) env->GetByteArrayElements(u, NULL);uint8_t *vArray = (uint8_t *) env->GetByteArrayElements(v, NULL);jint ySize = w * h;jint startx = (w - cw) / 2;jint starty = (h - ch) / 2;jint uvSize = (w >> 1) * (h >> 1);uint8_t *y_src = srcArray;uint8_t *uv_src = srcArray + ySize;uint8_t *y_transit = transitArray;uint8_t *u_transit = transitArray + ySize;uint8_t *v_transit = transitArray + ySize + uvSize;uint8_t *y_dst = dstArray;uint8_t *u_dst = dstArray + (cw * ch);uint8_t *v_dst = dstArray + (cw * ch) + ((cw * ch) >> 2);libyuv::NV21ToI420(y_src, w, uv_src, w, y_transit, w, u_transit, w >> 1, v_transit, w >> 1, w,h);libyuv::ConvertToI420(transitArray, ySize + uvSize, y_dst, cw, u_dst, cw >> 1,v_dst, cw >> 1,startx, starty, w, h, cw, ch, libyuv::kRotate0, libyuv::FOURCC_I420);memcpy(yArray, y_dst, cw * ch);memcpy(uArray, u_dst, (cw * ch) >> 2);memcpy(vArray, v_dst, (cw * ch) >> 2);releaseByteArray(env, transitJbyteArray, transitArray, 0);releaseByteArray(env, src, srcArray, 0);releaseByteArray(env, dst, dstArray, 0);releaseByteArray(env, y, yArray, 0);releaseByteArray(env, v, vArray, 0);releaseByteArray(env, u, uArray, 0);
} ;/** Class:     media_jni_NativeMethod* Method:    nv21CutterToNv12* Signature: ([B[BIIII)V*/
JNIEXPORT void JNICALL Java_media_jni_NativeMethod_nv21CutterToNv12(JNIEnv *env, jobject job, jbyteArray src, jbyteArray dst, jint cw, jint ch, jint w,jint h, jbyteArray y,jbyteArray u, jbyteArray v) {uint8_t *srcArray = (uint8_t *) env->GetByteArrayElements(src, 0);uint8_t *dstArray = (uint8_t *) env->GetByteArrayElements(dst, 0);uint8_t *yArray = (uint8_t *) env->GetByteArrayElements(y, NULL);uint8_t *uArray = (uint8_t *) env->GetByteArrayElements(u, NULL);uint8_t *vArray = (uint8_t *) env->GetByteArrayElements(v, NULL);jbyteArray transitJbyteArray = env->NewByteArray(w * h * 3 / 2);uint8_t *transitArray = (uint8_t *) env->GetByteArrayElements(transitJbyteArray, 0);jint ySize = w * h;jint startx = (w - cw) / 2;jint starty = (h - ch) / 2;jint uvSize = (w >> 1) * (h >> 1);uint8_t *y_src = srcArray;uint8_t *uv_src = srcArray + ySize;uint8_t *y_transit = transitArray;uint8_t *u_transit = transitArray + ySize;uint8_t *v_transit = transitArray + ySize + uvSize;uint8_t *y_dst = dstArray;uint8_t *uv_dst = dstArray + (cw * ch);libyuv::NV21ToI420(y_src, w, uv_src, w, y_transit, w, u_transit, w >> 1, v_transit, w >> 1, w,h);libyuv::ConvertToI420(transitArray, ySize + uvSize, yArray, cw, uArray, cw >> 1,vArray, cw >> 1,startx, starty, w, h, cw, ch, libyuv::kRotate0, libyuv::FOURCC_I420);libyuv::I420ToNV12(yArray, cw, uArray, cw >> 1, vArray, cw >> 1, y_dst, cw,uv_dst, cw, cw, ch);releaseByteArray(env, transitJbyteArray, transitArray, 0);releaseByteArray(env, src, srcArray, 0);releaseByteArray(env, dst, dstArray, 0);releaseByteArray(env, y, yArray, 0);releaseByteArray(env, v, vArray, 0);releaseByteArray(env, u, uArray, 0);
} ;
#ifdef __cplusplus
}
#endif
#endif以上就是jni层全部的代码基本上只涉及到数据转换和libyuv库的调用没啥太不好理解的地方。里面有注释旋转的代码有兴趣或者需求的童鞋可以恢复过来玩一玩。

这是yuv jni的目录。

这篇关于关于libyuv格式转换、图像裁剪说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/493507

相关文章

idea报错java: 非法字符: ‘\ufeff‘的解决步骤以及说明

《idea报错java:非法字符:‘ufeff‘的解决步骤以及说明》:本文主要介绍idea报错java:非法字符:ufeff的解决步骤以及说明,文章详细解释了为什么在Java中会出现uf... 目录BOM是什么?1. BOM的作用2. 为什么会出现 \ufeff 错误?3. 如何解决 \ufeff 问题?最

python+OpenCV反投影图像的实现示例详解

《python+OpenCV反投影图像的实现示例详解》:本文主要介绍python+OpenCV反投影图像的实现示例详解,本文通过实例代码图文并茂的形式给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一、前言二、什么是反投影图像三、反投影图像的概念四、反向投影的工作原理一、利用反向投影backproj

使用Python开发Markdown兼容公式格式转换工具

《使用Python开发Markdown兼容公式格式转换工具》在技术写作中我们经常遇到公式格式问题,例如MathML无法显示,LaTeX格式错乱等,所以本文我们将使用Python开发Markdown兼容... 目录一、工具背景二、环境配置(Windows 10/11)1. 创建conda环境2. 获取XSLT

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

Android使用ImageView.ScaleType实现图片的缩放与裁剪功能

《Android使用ImageView.ScaleType实现图片的缩放与裁剪功能》ImageView是最常用的控件之一,它用于展示各种类型的图片,为了能够根据需求调整图片的显示效果,Android提... 目录什么是 ImageView.ScaleType?FIT_XYFIT_STARTFIT_CENTE

Java对象转换的实现方式汇总

《Java对象转换的实现方式汇总》:本文主要介绍Java对象转换的多种实现方式,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java对象转换的多种实现方式1. 手动映射(Manual Mapping)2. Builder模式3. 工具类辅助映

python实现svg图片转换为png和gif

《python实现svg图片转换为png和gif》这篇文章主要为大家详细介绍了python如何实现将svg图片格式转换为png和gif,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录python实现svg图片转换为png和gifpython实现图片格式之间的相互转换延展:基于Py

使用Python实现图像LBP特征提取的操作方法

《使用Python实现图像LBP特征提取的操作方法》LBP特征叫做局部二值模式,常用于纹理特征提取,并在纹理分类中具有较强的区分能力,本文给大家介绍了如何使用Python实现图像LBP特征提取的操作方... 目录一、LBP特征介绍二、LBP特征描述三、一些改进版本的LBP1.圆形LBP算子2.旋转不变的LB

HTML5表格语法格式详解

《HTML5表格语法格式详解》在HTML语法中,表格主要通过table、tr和td3个标签构成,本文通过实例代码讲解HTML5表格语法格式,感兴趣的朋友一起看看吧... 目录一、表格1.表格语法格式2.表格属性 3.例子二、不规则表格1.跨行2.跨列3.例子一、表格在html语法中,表格主要通过< tab

C#实现将Excel表格转换为图片(JPG/ PNG)

《C#实现将Excel表格转换为图片(JPG/PNG)》Excel表格可能会因为不同设备或字体缺失等问题,导致格式错乱或数据显示异常,转换为图片后,能确保数据的排版等保持一致,下面我们看看如何使用C... 目录通过C# 转换Excel工作表到图片通过C# 转换指定单元格区域到图片知识扩展C# 将 Excel