for (int w = 0; w < lineDataSize; w += 3) {
mirrorBgr24[lineStartIndex + w] = bgr24[lineStartIndex + lineDataSize - w - 3];
mirrorBgr24[lineStartIndex + w + 1] = bgr24[lineStartIndex + lineDataSize - w - 2];
mirrorBgr24[lineStartIndex + w + 2] = bgr24[lineStartIndex + lineDataSize - w - 1];
}
lineStartIndex += lineDataSize;
}
}
- 垂直镜像
void verticalMirrorBgr24(char *bgr24, char *mirrorBgr24, int width, int height) {
int lineDataSize = width * 3;
bgr24 += width * height * 3;
for (int h = 0; h < height; h++) {
memcpy(mirrorBgr24, bgr24, lineDataSize);
mirrorBgr24 += lineDataSize;
bgr24 -= lineDataSize;
}
}
由于NV21和NV12的结构只是UV数据位置相反,因此镜像NV21的代码同样适用于镜像NV12。
- 水平镜像
void horizontalMirrorNv21(char *nv21, char *mirrorNv21, int width, int height) {
int yLineStartIndex = 0;
int uvLineStartIndex = width * height;
for (int h = 0; h < height; h++) {
for (int w = 0; w < width; w += 2) {
mirrorNv21[yLineStartIndex + w] = nv21[yLineStartIndex + width - w];
mirrorNv21[yLineStartIndex + w + 1] = nv21[yLineStartIndex + width - w - 1];
if ((h & 1) == 0) {
mirrorNv21[uvLineStartIndex + w] = nv21[uvLineStartIndex + width - w];
mirrorNv21[uvLineStartIndex + w + 1] = nv21[uvLineStartIndex + width - w - 1];
}
}
yLineStartIndex += width;
if ((h & 1) == 0) {
uvLineStartIndex += width;
}
}
}
- 垂直镜像
void verticalMirrorNv21(char *nv21, char *mirrorNv21, int width, int height) {
int yLineDataSize = width;
int nv21Index = 0, mirrorNv21Index = 0;
nv21Index += width * height - yLineDataSize;
//mirror y
for (int h = 0; h < height; h++) {
memcpy(mirrorNv21 + mirrorNv21Index, nv21 + nv21Index, yLineDataSize);
mirrorNv21Index += yLineDataSize;
nv21Index -= yLineDataSize;
}
int uvLineDataSize = width;
//mirror uv
int uvHeight = height / 2;
//point to final line of uv
nv21Index += (width * height * 3 / 2 - uvLineDataSize);
for (int h = 0; h < uvHeight; h++) {
memcpy(mirrorNv21 + mirrorNv21Index, nv21 + nv21Index, uvLineDataSize);
mirrorNv21Index += uvLineDataSize;
nv21Index -= uvLineDataSize;
}
}
由于I420 和YV12的结构只是UV数据位置相反,因此镜像I420的代码同样适用于镜像YV12。
- 水平镜像
void horizontalMirrorI420(char *i420, char *mirrorI420, int width, int height) {
int yLineStartIndex = 0;
int uLineStartIndex = width * height;
int vLineStartIndex = width * height * 5 / 4;
for (int h = 0; h < height; h++) {
for (int w = 0; w < width; w += 2) {
mirrorI420[yLineStartIndex + w] = i420[yLineStartIndex + width - w];
mirrorI420[yLineStartIndex + w + 1] = i420[yLineStartIndex + width - w - 1];
if ((h & 1) == 0) {
mirrorI420[uLineStartIndex + (w >> 1)] = i420[uLineStartIndex + ((width - w) >> 1)];
mirrorI420[vLineStartIndex + (w >> 1) + 1] = i420[vLineStartIndex +
((width - w) >> 1) - 1];
}
}
yLineStartIndex += width;
if ((h & 1) == 0) {
uLineStartIndex += width >> 1;
vLineStartIndex += width >> 1;
}
}
}
- 垂直镜像
void verticalMirrorI420(char *i420, char *mirrorI420, int width, int height) {
int halfWidth = width / 2;
int yLineDataSize = width;
int i420Index = 0, mirrorI420Index = 0;
i420Index += width * height - yLineDataSize;
//mirror y
for (int h = 0; h < height; h++) {
memcpy(mirrorI420 + mirrorI420Index, i420 + i420Index, yLineDataSize);
mirrorI420Index += yLineDataSize;
i420Index -= yLineDataSize;
}
int uvLineDataSize = halfWidth;
//mirror uv
int uHeight = height / 2;
//point to final line of u
i420Index += (width * height * 5 / 4 + yLineDataSize - uvLineDataSize);
for (int h = 0; h < uHeight; h++) {
memcpy(mirrorI420 + mirrorI420Index, i420 + i420Index, uvLineDataSize);
mirrorI420Index += uvLineDataSize;
i420Index -= uvLineDataSize;
}
//point to final line of v
i420Index += width * height / 2;
int vHeight = height / 2;
for (int h = 0; h < vHeight; h++) {
memcpy(mirrorI420 + mirrorI420Index, i420 + i420Index, uvLineDataSize);
mirrorI420Index += uvLineDataSize;
i420Index -= uvLineDataSize;
}
}
YUYV的共用关系是每2个Y共用1组U和V,因此在水平镜像时需要每4个Y替换其中两个Y的顺序。而垂直镜像时,整行处理即可。
- 水平镜像
void horizontalMirrorYuyv(char *yuyv, char *mirrorYuyv, int width, int height) {
int lineStartIndex = 0;
int lineDataSize = width * 2;
for (int h = 0; h < height; h++) {
for (int w = 0; w < lineDataSize; w += 4) {
mirrorYuyv[lineStartIndex + w] = yuyv[lineStartIndex + lineDataSize - w - 2];
mirrorYuyv[lineStartIndex + w + 1] = yuyv[lineStartIndex + lineDataSize - w - 3];
mirrorYuyv[lineStartIndex + w + 2] = yuyv[lineStartIndex + lineDataSize - w - 4];
mirrorYuyv[lineStartIndex + w + 3] = yuyv[lineStartIndex + lineDataSize - w - 1];
}
lineStartIndex += lineDataSize;
}
}
学习福利
【Android 详细知识点思维脑图(技能树)】
其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
[外链图片转存中…(img-dqTS74us-1714956251090)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!