全景图片(鱼眼)的平面映射矫正

2023-11-11 19:59

本文主要是介绍全景图片(鱼眼)的平面映射矫正,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近做了一个全景图片平面映射的工作,就是将一个360度的全景球面照片映射到一个平面上,使之看上去没有变形。由于网上的一些鱼眼照片的校正程序不好用,自己通过球体的三角计算,找到了映射效果较好的方法。写博客以备忘。先看看效果。


 做完以后又进行了一些改造,使程序变成一个可变大小,可变角度的相对较通用的程序。本人能力偏弱,只能用笨办法想问题,不足之处敬请海涵。

主要的算法是这样的。当从平面转成球面的时候,需要从圆心引一条直线,通过平面图片的每一个像素,打到球面上,把这个平面的像素给球面即可。我想平面的映射应该就是反过程。关键是怎么求球面的点与平面的点之间的关系。因为虽然是球面的图片,但是只是视觉上是球面的,其图片结构还是二位的像素点。我们来看看具体的分析。


       图片上就是我手绘的一个示意图。将球面映射到平面时,需要新建一个平面图片。对平面的图片的横纵左边点循环,跑遍每一个点,做点与圆心o的连线,打到球面上,球面上点与球面图片的左右边界之间的弧长,就是我们要求的球面图片的横坐标。那么弧长如何计算呢?很显然就是使用圆心角的角度乘以球的半径。(因为过圆心,所以是大圆。半径就是把图片的宽度作为大圆周长,求出来的半径R)。圆心角如何求呢?我能想到的比较直接的办法是三角几何。图片上的A点就是我们的平面上任意一点,我们要求的是Ao与红线的夹角。那么就是Do与红线夹角减去Do与Ao夹角就是了。Do与红线夹角为平面图片长度的一般,假设是length/2。Do的长度可以用DCo这个直角三角形,通过勾股定理来求,值为sqrt(y*y+length/2*length/2)。红线与A所在水平直线的交点假设为E,那么ED就是length/2。所以DoE=arctan(ED/Do)。DoA=arctan((length/2-x)/Do)。那么我们求的AoE就是DoE-DoA=arctan(ED/Do)-arctan((length/2-x)/Do)。用这个角度直接乘以半径R就是横坐标了。当点在平面的右半部分时,把length/2-x变成x-length/2即可。

        纵坐标的求取跟这个很像,只需要求AoB的角度。通过BC与oC的直角三角形勾股定理求Bo。然后AoB就是arctan(AB/Bo)。球的这段弧长就是arctan(AB/Bo)*R但是还有一个问题,现在只是求的弧长,但是我们要求的是球面图片的纵坐标,因此需要用球面图片的高度的一般减去弧长。这样我们就求得了每个点对应的坐标。将图片显示出来即可。

        这个是基本的思路,程序和算法都很简单,但效果不错,也比较稳定。

       因为是360度的全景,因此显示完一部分后我们会想显示其他部分,我们将想显示的角度放到一个vector<float>d中,然后在循环中变化角度,把每个角度转换完的图片放到一个vector<IplImage*>results向量中,并显示出来。在使用vector<IplImage*>results向量,push_back时,遇到了麻烦,耽误了快一天的时间解决这个push_back会覆盖之前图像的问题,愁死我了。这个会在另外的一篇博客中提高。我的工程文件在资源中可以找到。

           还有一个问题是求出来的图像像素点是近似的,因此会模糊,需要处理一下,就是讲不是整数的点,通过上下左右四个点,求一下平均,图像会好很多。函数的主要部分是这样的:

[cpp]  view plain copy
  1. int comeon(IplImage* srcImg,vector<float>& directions, float angle, int xDim, int yDim,vector<IplImage*>& results)    
  2. {    
  3.         //srcImg是原图片,directions是需要的方向,angle是视角范围,xDim和yDim是映射图片大小,results中存放结果  
  4.     Mat src(srcImg);   
  5.     Mat img(xDim+1,yDim+1,CV_8UC3);  
  6.     float z=xDim/(2*tan(angle*3.1415926/360));  
  7.     TickMeter tm;       
  8.     tm.start();   
  9.     for(int i=0;i<directions.size();i++)  
  10.     {  
  11.           
  12.         for (int y=0; y<yDim; y++)    
  13.         {    
  14.   
  15.             uchar* P1  = img.ptr<uchar>(y);   
  16.             uchar* P0  = src.ptr<uchar>(y);         
  17.             for (int x=0; x<xDim; x++)    
  18.             {  
  19.                 float c;  
  20.                 if(y<=(int)(yDim/2))  
  21.                     c=change1(x,y,xDim*1.0/2.0,yDim*1.0/2.0,z);  
  22.                 else  
  23.                     c=change2(x,y,xDim*1.0/2.0,yDim*1.0/2.0,z);  
  24.                 int d=(int)c;  
  25.   
  26.                 P0  = src.ptr<uchar>(c);   
  27.               
  28.                 float a;  
  29.                 a=change(x,y,xDim*1.0/2.0,z);  
  30.                 a=a+directions[i];  
  31.   
  32.                 int   b=(int)a;  
  33.           
  34.   
  35.                 float B=P0[3*b]*(1-a+b)+P0[3*(b+1)]*(a-b);    
  36.                 float G=P0[3*b+1]*(1-a+b)+P0[3*(b+1)+1]*(a-b);    
  37.                 float R=P0[3*b+2]*(1-a+b)+P0[3*(b+1)+2]*(a-b);    
  38.   
  39.                 P0  = src.ptr<uchar>(c+1);  
  40.                 float B1=P0[3*b]*(1-a+b)+P0[3*(b+1)]*(a-b);    
  41.                 float G1=P0[3*b+1]*(1-a+b)+P0[3*(b+1)+1]*(a-b);    
  42.                 float R1=P0[3*b+2]*(1-a+b)+P0[3*(b+1)+2]*(a-b);  
  43.   
  44.                 B=B*(1-c+d)+B1*(c-d);  
  45.                 G=G*(1-c+d)+G1*(c-d);  
  46.                 R=R*(1-c+d)+R1*(c-d);  
  47.   
  48.                 P1[3*x] = (uchar)B;    
  49.                 P1[3*x+1] = (uchar)G;    
  50.                 P1[3*x+2] = (uchar)R;  
  51.             }  
  52.         }  
  53.   
  54.         IplImage *res;  
  55.         res =(_IplImage*) malloc(sizeof(_IplImage));  
  56.         *res=IplImage(img);  
  57.   
  58.         IplImage* tempimg = (IplImage*)cvClone(res);  
  59.         results.push_back(tempimg);  
  60.   
  61.     }  
  62.     tm.stop();    
  63.     cout<<"process time="<<tm.getTimeMilli()<<endl;   
  64.     return 1;  
  65. }   
横坐标求取代码如下:

[cpp]  view plain copy
  1. float change(int x,int y,float xDim,float z)  
  2. {  
  3.     float tt=(xDim-x)/z;  
  4.     float l=120*3.1415926/360-atan(tt);  
  5.     float result=l*r;  
  6.     return result;    
  7. }  





这篇关于全景图片(鱼眼)的平面映射矫正的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python实现精准提取 PDF中的文本,表格与图片

《Python实现精准提取PDF中的文本,表格与图片》在实际的系统开发中,处理PDF文件不仅限于读取整页文本,还有提取文档中的表格数据,图片或特定区域的内容,下面我们来看看如何使用Python实... 目录安装 python 库提取 PDF 文本内容:获取整页文本与指定区域内容获取页面上的所有文本内容获取

Python基于微信OCR引擎实现高效图片文字识别

《Python基于微信OCR引擎实现高效图片文字识别》这篇文章主要为大家详细介绍了一款基于微信OCR引擎的图片文字识别桌面应用开发全过程,可以实现从图片拖拽识别到文字提取,感兴趣的小伙伴可以跟随小编一... 目录一、项目概述1.1 开发背景1.2 技术选型1.3 核心优势二、功能详解2.1 核心功能模块2.

Go语言如何判断两张图片的相似度

《Go语言如何判断两张图片的相似度》这篇文章主要为大家详细介绍了Go语言如何中实现判断两张图片的相似度的两种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 在介绍技术细节前,我们先来看看图片对比在哪些场景下可以用得到:图片去重:自动删除重复图片,为存储空间"瘦身"。想象你是一个

使用Python实现base64字符串与图片互转的详细步骤

《使用Python实现base64字符串与图片互转的详细步骤》要将一个Base64编码的字符串转换为图片文件并保存下来,可以使用Python的base64模块来实现,这一过程包括解码Base64字符串... 目录1. 图片编码为 Base64 字符串2. Base64 字符串解码为图片文件3. 示例使用注意

Nginx Location映射规则总结归纳与最佳实践

《NginxLocation映射规则总结归纳与最佳实践》Nginx的location指令是配置请求路由的核心机制,其匹配规则直接影响请求的处理流程,下面给大家介绍NginxLocation映射规则... 目录一、Location匹配规则与优先级1. 匹配模式2. 优先级顺序3. 匹配示例二、Proxy_pa

c/c++的opencv实现图片膨胀

《c/c++的opencv实现图片膨胀》图像膨胀是形态学操作,通过结构元素扩张亮区填充孔洞、连接断开部分、加粗物体,OpenCV的cv::dilate函数实现该操作,本文就来介绍一下opencv图片... 目录什么是图像膨胀?结构元素 (KerChina编程nel)OpenCV 中的 cv::dilate() 函

使用Python实现调用API获取图片存储到本地的方法

《使用Python实现调用API获取图片存储到本地的方法》开发一个自动化工具,用于从JSON数据源中提取图像ID,通过调用指定API获取未经压缩的原始图像文件,并确保下载结果与Postman等工具直接... 目录使用python实现调用API获取图片存储到本地1、项目概述2、核心功能3、环境准备4、代码实现

Python+PyQt5实现文件夹结构映射工具

《Python+PyQt5实现文件夹结构映射工具》在日常工作中,我们经常需要对文件夹结构进行复制和备份,本文将带来一款基于PyQt5开发的文件夹结构映射工具,感兴趣的小伙伴可以跟随小编一起学习一下... 目录概述功能亮点展示效果软件使用步骤代码解析1. 主窗口设计(FolderCopyApp)2. 拖拽路径

Java实现图片淡入淡出效果

《Java实现图片淡入淡出效果》在现代图形用户界面和游戏开发中,**图片淡入淡出(FadeIn/Out)**是一种常见且实用的视觉过渡效果,它可以用于启动画面、场景切换、轮播图、提示框弹出等场景,通过... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细

Java如何根据文件名前缀自动分组图片文件

《Java如何根据文件名前缀自动分组图片文件》一大堆文件(比如图片)堆在一个目录下,它们的命名规则遵循一定的格式,混在一起很难管理,所以本文小编就和大家介绍一下如何使用Java根据文件名前缀自动分组图... 目录需求背景分析思路实现代码输出结果知识扩展需求一大堆文件(比如图片)堆在一个目录下,它们的命名规