使用切线方法,对切线方向上的边缘进行强化:

参考连接:图像锐化和边缘检测

代码:

		//在种子点方向上寻找合适的梯度,用于寻找边缘
//对low_Gray, high_gray之间的点寻找边缘
void FindBestGradient(
cv::Mat &_src, cv::Mat &_dst,
cv::Point2f &seed,
float low_Gray,
float high_gray,
int aperture_size, bool oPenEnhence )
{
//角度矩阵
cv::Mat df = cv::Mat::zeros( _src.rows,_src.cols, CV_32FC1 );
//梯度矩阵
cv::Mat dg = cv::Mat::zeros( _src.rows,_src.cols, CV_32FC1 );
//原始图像
cv::Mat ds = _src.clone();
//目标图像 uchar型
cv::Mat dd = _src.clone(); //1.根据角度计算梯度//得到梯度矩阵
//使用N*1的算子
int n = aperture_size;//必须为奇数 //对每个柱进行初始化
//搜索柱:在射线方向上搜索l_Search 个像素;宽度为
int l_Search = n;
int w_Search = 1;
std::vector<std::vector<std::pair<cv::Point ,float> > > beam;
beam.resize( l_Search );
for (int i=0;i< beam.size();++i)
{
beam[i].resize(w_Search);
}//初始化柱 //设定系数//生成模板
double gap = 2.0/ (n-1);
std::vector< double > mask(l_Search);
for (int i=0;i< mask.size();++i)
{
mask[i] = -1 + i*gap ;
} //2.生成角度图像
//在射线方向上寻找//方法不是太好,但是没有寻找到简单有效的方法
for ( int y=0 ;y< ds.rows;++y )
{
float* ptr = (float*)( df.data + y * df.step);
unsigned char* pS = ( unsigned char* )( ds.data + y * ds.step);
for ( int x=0; x< ds.cols; ++x )
{
//计算角度
if ( (int)(*pS) > low_Gray && (int)(*pS) <high_gray )
{
*ptr = (float)(cvWish::cosCv(seed,cv::Point2f( x,y ) ) );
}
else
{
*ptr = 0.00000000000f;
}
++ptr;
++pS;
}
} //计算差值-导数
for (int y=0 ;y< ds.rows;++y)
{
float* pf = (float*)( df.data + y * df.step);
float* pg = (float*)( dg.data + y * dg.step);
unsigned char* pd = (unsigned char*)( dd.data + y * dd.step); for (int x=0;x< ds.cols;++x )
{
//计算角度
if ( abs((float)(*pf)) > 0.00000001 )
{
//cvWish::BeamInit(l_Search,w_Search,cv::Point2f( x,y ),df.at<float >(y,x),beam,0);//0表示从中部开始搜索
cvWish::BeamInit(l_Search,w_Search,cv::Point2f( x,y ), *pf ,beam,0);//0表示从中部开始搜索
cvWish::BeamNormal(dg.cols, dg.rows , beam); *pg = 0;
for ( int k =0; k< l_Search; ++k ){
*pg += (float)( mask[k]* ds.at<unsigned char>(beam[k][0].first.y,beam[k][0].first.x) );
}
int s = abs ( ( (int)(*pg ) )%255 ) ;
*pd = (unsigned char) (s);
}
else
{
*pd = (unsigned char) (0);
} ++pf;
++pg;
++pd;
}
} cv::Mat edgeMat = dd;
cv::Mat angleMat= df;
int maskSize = 5;
if ( oPenEnhence )
{
int num = 1;
for (int i=0;i< num;++i)
{
EnhanceEdgeByTangent( edgeMat ,angleMat, maskSize);
} }
else
{
}
_dst = edgeMat.clone();
return;
}

边缘强化函数:

//使用边缘增强--沿切线方向增强
//方向性,边缘限制
void EnhanceEdgeByTangent( cv::Mat &edgeMat ,cv::Mat &angleMat, int maskSize)
{
cv::Mat ds = edgeMat;
cv::Mat dd = edgeMat.clone(); cv::Mat df = angleMat;
cv::Mat dg = angleMat.clone();//导数图,最终转化为 灰度图 dd const int l_Search = maskSize;
const int w_Search = 1;
//初始化柱
std::vector<std::vector<std::pair<cv::Point ,float> > > beam;
beam.resize( l_Search );
for (int i=0;i< beam.size();++i)
{
beam[i].resize(w_Search);
}//初始化柱 //设定系数//生成模板
double gap = 2.0/ ( l_Search - 1);
std::vector< double > mask(l_Search);
for (int i=0;i< mask.size();++i)
{
mask[i] =abs( 1- abs( -1 + i*gap ) );
} //强化边缘
for (int y=0 ;y< ds.rows;++y)
{
float* pf = (float*)( df.data + y * df.step);
float* pg = (float*)( dg.data + y * dg.step);
unsigned char* pd = (unsigned char*)( dd.data + y * dd.step); for ( int x=0; x< ds.cols; ++x )
{
//计算角度
if ( abs((float)(*pf)) > 0.00000001 )
{
float angle = *pf + PI_1_2 ;//切线方向,加 PI_1_2
angle = angle>=PI_4_2? angle - PI_4_2:angle; cvWish::BeamInit( l_Search, w_Search, cv::Point2f( x,y ), angle , beam, 0 );
cvWish::BeamNormal(dg.cols, dg.rows , beam); *pg = 0;
const int gl= ds.at<unsigned char>(y,x) ;//当前像素值
int vvv = dd.at<unsigned char>(y,x);
for ( int k =0; k< l_Search; ++k ){
vvv += gl* mask[k];
}
if ( vvv>255 )
{
vvv=255 ;
}
dd.at<unsigned char>(y,x) = vvv; //*pd = (unsigned char) (s);
}
else
{
*pd = (unsigned char) (0);
}
++pf;
++pg;
++pd;
}
} edgeMat = dd.clone();
return ;
}

图片效果:

最新文章

  1. 安装SQL Developer,连接Oracle 12c,创建新用户
  2. 利用QJSON将FDQuery转成JSON串
  3. shell中bc expr [ ] (( ))的使用方法
  4. Echarts-画柱状,折线图
  5. [收藏夹整理]OpenCV部分
  6. Samsung_tiny4412(驱动笔记01)----linux 3.5,U-Boot,Busybox,SD卡启动环境搭建
  7. 监控mysql索引使用效率的脚本
  8. java实现的Trie树数据结构
  9. 推荐五个最好的MySQL GUI工具
  10. 关于easyui的tab,layout,datagrid嵌套的问题
  11. 实例学习SSIS(五)--理论介绍SSIS
  12. Head First设计模式之原型模式
  13. CentOS安装Nginx 以及日志管理
  14. 编译android --system,framework
  15. Python 3.0 写日志时出现乱码
  16. 改善C#程序的建议9:使用Task代替ThreadPool和Thread
  17. TMS320VC5509驱动LCD1602
  18. Block修改变量容易被忽略的方法
  19. VDI数据恢复
  20. phpcms 模板学习

热门文章

  1. Java 下实现Cache
  2. Python 1 初识python
  3. 暑假集训D17总结
  4. MySQL 调优 —— Using filesort
  5. 最经典的SDK程序结构 HelloWin
  6. luogu2149 [SDOI2009] Dlaxia的路线
  7. c# winform 多条件查找 外加网络人才回答
  8. CodeForces - 789D Weird journey
  9. [BZOJ 1741] Asteroids
  10. 基于Spark ML的Titanic Challenge (Top 6%)