Android openh264 编解码重点记录
2016-03-16 17:05
921 查看
编译工具:CYGWIN
只要进入到openh264根目录 就能够直接根据github提供的编译步骤进行编译,之后会获得openh264相关的动态和静态库.我的openh264是15年8月下载的下载地址
编码部分:
1 从android获取摄像头数据使用的格式是NV21(yuv420sp),由于手机拍摄默认是横屏,所以获取到数据需要将数据进行旋转(是否有更好的方式处理 之后再研究).顺时针旋转90°,从stackoverflow上找的
private byte[] rotateYUV420Degree90(byte[] data, int imageWidth, int imageHeight) { byte[] yuv = new byte[imageWidth * imageHeight * 3 / 2]; // Rotate the Y luma int i = 0; for (int x = 0; x < imageWidth; x++) { for (int y = imageHeight - 1; y >= 0; y--) { yuv[i] = data[y * imageWidth + x]; i++; } } // Rotate the U and V color components i = imageWidth * imageHeight * 3 / 2 - 1; for (int x = imageWidth - 1; x > 0; x = x - 2) { for (int y = 0; y < imageHeight / 2; y++) { yuv[i] = data[(imageWidth * imageHeight) + (y * imageWidth) + x]; i--; yuv[i] = data[(imageWidth * imageHeight) + (y * imageWidth) + (x - 1)]; i--; } } return yuv; }
注意:旋转之后的宽高在放进去编码时需要调换过来,不然等到后面解码出来会出现显示的长宽反了
2 openh264编码数据只接收yuv i420(yuv420p)数据(别的暂时还没试),所以在这里进行了YUV420SP转YUV420P的步骤,代码如下:
public byte[] nv21ToI420(byte[] yuv420sp, int width, int height) { int framesize = width * height; byte[] yuv420 = new byte[framesize * 3 / 2]; int i = 0, j = 0; // copy y for (i = 0; i < framesize; i++) { yuv420[i] = yuv420sp[i]; } i = 0; for (j = 0; j < framesize / 2; j += 2) { yuv420[i + framesize] = yuv420sp[j + framesize]; yuv420[i + framesize * 5 / 4] = yuv420sp[j + 1 + framesize]; i++; } return yuv420; }
这样就能编码了
解码部分
1 使用官方提供的编解码步骤,解码出来的数据重点,需要将Y、U、V三部分进行组合成一个YUV420数据unsigned char* buf = new unsigned char[m_iWidth * m_iHeight * 3 / 2]; int a = 0; for (int i = 0; i < m_iHeight; i++) { memcpy(buf + a, data[0] + i * bufInfo.UsrData.sSystemBuffer.iStride[0], m_iWidth); a += m_iWidth; } for (int i = 0; i < m_iHeight / 2; i++) { memcpy(buf + a, data[1] + i * bufInfo.UsrData.sSystemBuffer.iStride[1], m_iWidth / 2); a += m_iWidth / 2; } for (int i = 0; i < m_iHeight / 2; i++) { memcpy(buf + a, data[2] + i * bufInfo.UsrData.sSystemBuffer.iStride[1], m_iWidth / 2); a += m_iWidth / 2; }
2 之后将这个YUV数据进行转换成ARGB以便Android显示
jbyte* yuv420p = reinterpret_cast<jbyte*>(buf); jint* rgb = new jint[frameSize]; int i = 0, j = 0, yp = 0; int up = 0, vp = 0, u = 0, v = 0; for (j = 0, yp = 0; j < m_iHeight; j++) { up = frameSize + (j >> 2) * m_iWidth; vp = frameSize * 5 / 4 + (j >> 2) * m_iWidth; u = 0; v = 0; for (i = 0; i < m_iWidth; i++, yp++) { int y = (0xff & ((int) yuv420p[yp])); if (y < 0) y = 0; if ((i & 1) == 0) { u = (0xff & yuv420p[up++]); v = (0xff & yuv420p[vp++]); } int y1192 = y_table[y]; int r = r_yv_table[y][u]; int g = (y1192 - g_v_table[u] - g_u_table[v]); int b = b_yu_table[y][v]; if (g < 0) g = 0; else if (g > 262143) g = 262143; rgb[yp] = 0xff000000 | ((r << 6) & 0xff0000) | ((g >> 2) & 0xff00) | ((b >> 10) & 0xff); } } env->SetIntArrayRegion(result, 0, frameSize, rgb); delete[] rgb;这段代码由网上找来的,稍微修改,改成I420转ARGB
解码出来之后就可以到Android显示了,显示用的是opengl,代码太多就不放出来了,网上很多。
============================================================================================
2016-5-18
yuv的显示直接用opengl显示了,而不必转成rgb再去显示了
============================================================================================
2016-10-21
一段时间没时间去研究视频显示,现在要用的时候发现一个问题:前置摄像头和后置摄像头的视频排版不一样
前置摄像头和后置摄像头区别
1.preview输出的data数据:
以横屏为标准
前置摄像头是反方想的,即正上方向右,后置摄像头和视频显示出来的图像是正的 即正上方向左
2.旋转的时候就需要注意:
后置摄像头是正常旋转,前置摄像头需要左右调换再旋转
ps:记住前置摄像头是相反的
3
前置摄像头和后置摄像头区别
1.preview输出的data数据:
前置摄像头是反方想的,即正上方向右,后置摄像头和视频显示出来的图像是正的 即正上方向左
2.旋转的时候就需要注意:
后置摄像头是正常旋转,前置摄像头需要左右调换再旋转
3.yuv颜色显示:
后置摄像头正常Y-U-V 而前置摄像头是Y-V-U
相关文章推荐
- 使用C++实现JNI接口需要注意的事项
- Android IPC进程间通讯机制
- Android Manifest 用法
- [转载]Activity中ConfigChanges属性的用法
- Android之获取手机上的图片和视频缩略图thumbnails
- Android之使用Http协议实现文件上传功能
- Android学习笔记(二九):嵌入浏览器
- android string.xml文件中的整型和string型代替
- i-jetty环境搭配与编译
- android之定时器AlarmManager
- android wifi 无线调试
- Android Native 绘图方法
- Android java 与 javascript互访(相互调用)的方法例子
- android 代码实现控件之间的间距
- android FragmentPagerAdapter的“标准”配置
- Android"解决"onTouch和onClick的冲突问题
- android:installLocation简析
- android searchView的关闭事件
- SourceProvider.getJniDirectories