需求说明

客户端接收到服务器传送过来的图像数据,客户端通过对图像进行旋转和反转操作. 然后把这个旋转和反转的数据上传到服务器. 客户端在接收图像的时候, 也会下载以前的旋转和反转参数, 然后客户端根据这些数据对图形自动进行旋转和反转. 这是一个图像的矫正问题. 有大量的图像, 图像的有可能是旋转的,反转的.使用之前需要旋正和反正.

问题分析

初步分析,感觉很简单. 我只要记录旋转角度和反转状态,上传到服务器, 然后下载,根据参数做变化. 不知道你考虑到 旋转和反转的顺序问题木有?, 对于一张图像, 先旋转,后反转 和 先反转,后旋转是不同的. 还需要考虑的是, 把这2类型的参数上传到服务器, 下次, 下载的时候, 我是先用旋转参数好呢? 还是先用反转参数呢? 一些问题是存在数学规律的, 找到规律就可以迎刃而解.

旋转分为 顺时针和逆时针, 每次90度. 顺时针则+90, 逆时针则-90

反转分为 水平反转和垂直反转, 以及不反转 3种状态, 我们使用枚举类型 H_FILP,V_FLIP, NO_FLIP

对于反转, 我们得到如下规律:

  1. 同一方向翻转两次, 相当于没有翻转, 也就是 水平反转2次,等于没有反转. 垂直相同.
  2. 两次不同的翻转,相当于图像旋转180度. 也就是, 水平反转一次, 然后垂直反转一次, 等价于图像(顺时针)旋转了180度
  3. 当上一次没有旋转的时候, 这一次是H,就是H,是V就是V.

对于旋转, 我们得到如下规律:

  1. 分为顺时针(左旋), 逆时针(右旋), 每次旋转分为90度.
  2. 旋转的时候, 如果存在翻转, 那么翻转的方向就要改变.

伪代码实现


/** 反转状态 */
enum FLIP_STATE
{
NO_FLIP = 0, ///< 没有反转
H_FLIP, ///< 水平反转
V_FLIP ///< 垂直反转
}; /** 主要使用的变量(可以是类的成员变量) */
int m_nOldRotate; ///< Old 纪录初始旋转的参数,图像顺时针旋转0,90,180,270, 用于旋转和翻转的还原
FLIP_STATE m_oldFlipState;
int m_nRotate; ///< 记录旋转的参数,图像顺时针旋转0,90,180,270
FLIP_STATE m_flipState; m_imShowData: 就来图像的信息类的对象 /** 水平反转事件 */
void OnHorFlip()
{
/** 反转前 */ /** 如果上一次状态是没有反转, 那么就把反转状态设置为水平反转 */
if ( NO_FLIP == m_imShowData.m_flipState ){m_imShowData.m_flipState = H_FLIP;} /** 如果上一次状态是水平反转, 那么2次水平反转, 就是没有反转 */
else if ( H_FLIP == m_imShowData.m_flipState ){m_imShowData.m_flipState = NO_FLIP;} /** 如果上一次状态是垂直反转, 也就是上一次是垂直,这次我要水平了, 那么两次不同方向的反转,
等价于 旋转180度, 反转设置为没有反转 */
else if ( V_FLIP == m_imShowData.m_flipState )
{
m_imShowData.m_nRotate = (m_imShowData.m_nRotate + 180) % 360;
m_imShowData.m_flipState = NO_FLIP;
} /** 将图像水平旋转 */
OperateImage(HOR_FLIP);
} /** 垂直反转事件 */
void OnVerFlip()
{
/** 垂直反转和水平是类似的逻辑, 请自行思考. */
if ( NO_FLIP == m_imShowData.m_flipState ){m_imShowData.m_flipState = V_FLIP;}
else if ( V_FLIP == m_imShowData.m_flipState ){m_imShowData.m_flipState = NO_FLIP;}
else if ( H_FLIP == m_imShowData.m_flipState )
{
m_imShowData.m_nRotate = (m_imShowData.m_nRotate + 180) % 360;
m_imShowData.m_flipState = NO_FLIP;
}
OperateImage(VER_FLIP);
} /** 逆时针旋转事件 */
void OnLRotate()
{
if ( m_imShowData.m_flipState != NO_FLIP )
{
m_imShowData.m_flipState = (m_imShowData.m_flipState == H_FLIP) ? V_FLIP : H_FLIP;
}
m_imShowData.m_nRotate = (m_imShowData.m_nRotate - 90) % (-360);
OperateImage(LEFT_ROTATE);
} /** 顺时针旋转事件 */
void OnRRotate()
{
/** 如果存在反转, 则反之方向,要发生变化 */
if ( m_imShowData.m_flipState != NO_FLIP )
{
m_imShowData.m_flipState = (m_imShowData.m_flipState == H_FLIP) ? V_FLIP : H_FLIP;
}
/** 旋转参数累加, 不做解释 */
m_imShowData.m_nRotate = (m_imShowData.m_nRotate + 90) % 360;
OperateImage(RIGHT_ROTATE);
} /**
此函数只用于从引擎获得旋转和翻转数据以后,对原始图像进行调整使用:
获得的 旋转和翻转两个参数, 先旋转后翻转和先翻转后旋转着两种情况是不一样的.
经过测试, 应该先旋转后翻转.
*/
void SetImgRotateFlip( ImgShowData &img )
{
int nTimes = img.m_nRotate / 90;
while(nTimes > 0)
{
OperateImage(RIGHT_ROTATE);
nTimes--;
} if( img.m_flipState == H_FLIP ){OperateImage(HOR_FLIP);}
else if( img.m_flipState == V_FLIP ){OperateImage(VER_FLIP);}
} /** 还原很简单, 重新赋值, 调用上面的函数即可 */
void OnRevert()
{
m_imShowData.m_nRotate = m_imShowData.m_nOldRotate;
m_imShowData.m_flipState = m_imShowData.m_oldFlipState;
SetImgRotateFlip(m_imShowData);
} /*
* 最后说一下,上传, 直接上传参数数值即可. 只是,服务器对于旋转存储的都是顺时针的数值, 也就是0,90,180,270.
* 客户端,因为存在逆时针, 所以,最后需要把逆时针的数值, 转化到顺时针, 即,逆时针的补角即可.
*/

小结

在一个坑蹲的时间长了. 这块儿地方, 才会被你改变.


How.To.Process.Image.Infomation.Of.Rotate.And.Flip.From.Server的更多相关文章

  1. Contents Of My Blogs

    C++ How To Use Goto? Preprocessing Directive std::array std::deque std::forward_list std::map std::m ...

  2. Blocked Process Report

    当同个对象上有互斥的锁存在时,查询需要等待很长时间,我们是否可以收到来自SQL Server提醒?答案是可以的,做法非常简单,因为SQL Server为你提供了称为Blocked Process Re ...

  3. android.process.acore和system进程

    从源码看来,android.process.acore进程应该是一些基本功能的载入程序. android-4.3_r2.2中,它包括以下项目: 1.UserDictionaryProvider < ...

  4. 使用Java监控工具出现 Can't attach to the process

    问题重现 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 ➜ jinfo -flags 3032 Attaching ...

  5. Server Process

    1.client进行update操作后.其它是怎么协作的? Client进行update操作之后,是由Server Process真正完毕的,分以下几步: 1).须要更新的数据在Data buffer ...

  6. 05 Oracle process

    本章提要----------------------------------------------所有的 process 都是在 PGA 内(memory)server process: 与 cli ...

  7. 多并发编程基础 之进程 Process

    原贴  https://www.cnblogs.com/gbq-dog/p/10299663.html 1. 进程的理论知识 1.1 操作系统的背景知识 顾名思义,进程即正在执行的一个过程.进程是对正 ...

  8. 【oracle学习笔记02】Oracle Architecture —— Process Structure

    Oracle中有三类进程: 1 User Process 2 Server Process Server Process is a program that directly interacts wi ...

  9. canvas/CSS实现仪表盘效果

    手机上看比较虚 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <t ...

随机推荐

  1. gitbook安装与使用之windows下搭建gitbook平台

    最近需要在GitBook中去阅读电子书 安装nodejs cnpm安装gitbook 解压书籍文件,并cd到书籍文件目录 gitbook serve 浏览器访问localhost:4000 先在win ...

  2. 打印自己的C代码

    #include "stdio.h" ,,,,p,,);;}

  3. 给mac配置adb 路径

    给mac配置adb 路径 (1)找到对应的adb 所在路径 /Users/***/Library/Android/sdk/platform-tools (2)启动终端,输入 cd $Home (3)输 ...

  4. LeetCode刷刷记录

    一遍考研,一遍还是要刷刷题.感觉自己的时间安排的不是很好,还是要抓紧自己的日常时间,当然,也要练练刷题的手感. 1.第一题就两重循环找到索引就OK,因为是无序的,所以就不能用二分来查找,题目中每个数的 ...

  5. 求50-100内的素数(java)

    实现代码: public class sushu { public static void main(String[] args) { for(int i=50 ; i<=100; i++){ ...

  6. HDU5509 : Pattern String

    只要求出两个字符串的最小表示,然后就可以判断是否循环同构. 枚举最小表示的开头在哪个位置,然后求出Hash值,如果两个串的Hash值集合有交,那么说明循环同构. 因为串经过压缩,原串的长度很大,不能直 ...

  7. Attention:本博客暂停更新

    Attention:本博客暂停更新 2016年11月17日08:33:09 博主遗产 http://www.cnblogs.com/radiumlrb/p/6033107.html Dans cett ...

  8. [BZOJ2072][POI2004] MOS过桥

    Description 一个夜晚一些旅行者想要过桥. 他们只有一个火把. 火把的亮光最多允许两个旅行者同时过桥. 没有火把或者多于2个人则不能过桥.每个旅行者过桥都需要特定的时间, 两个旅行者同时过桥 ...

  9. js排序算法总结——冒泡,快速,选择,插入,希尔,归并

    相信排序是任何一个程序猿都会用到的东西,今天简单总结记录下常见的排序算法. 一.冒泡排序 说起冒泡排序,可能每个人都不会陌生,实现思路相当简单明了,就是不停的对数组进行两两比较,将较大(较小)的一项放 ...

  10. mac下配置xampp多端口

    首先下载并安装完XAMPP软件. 第一步: 打开XAMPP安装目录,找到配置文件. 如:/Applications/XAMPP/etc/httpd.conf 打开后查找 Listen 80 会看到以下 ...