图像处理之积分图应用二(快速边缘保留滤波算法)

图像处理之积分图应用二(快速边缘保留滤波算法)

一:基本原理

传统的图像边缘保留滤波算法-如高斯双边模糊、Mean-Shift模糊等计算复杂、效率比较低,虽然有各种手段优化或者快速计算方法,当时算法相对一般码农来说理解起来比较费劲,不是一个的选择,而通过积分图像实现局部均方差的边缘保留模糊算法,计算简单而且可以做到计算量跟半径无关、跟上面提到两种边缘保留滤波(EPF)算法效率高很多。首先局部均方差滤波中计算局部均值的公式如下:

当边缘很弱的时候系数K趋近于0、该点的矫正之后的像素值就接近平均值。而当边缘很强的时候系数K趋近于1、该点的模糊之后的像素值就接近等于输入像素值。上述计算中最中意的是窗口内像素的均值与方差,计算均值可以根据积分图像很容易得到,而计算方差根据一系列的数学推导可以得到如下:


就是说可以根据积分图像通过常量次数的计算得到局部的均值与方差,让这种情况下的滤波变成一个常量时间完成的操作与窗口半径大小无关。

二:算法流程

1. 根据输入的图像计算得到积分图像,参见《图像处理之积分图像算法》

2. 根据输入的半径大小计算窗口内像素均值与方差、计算得到每个像素新的像素值

3. 循环每个个像素,重复第2步计算,得到最终的局部均方差滤波图像

三:代码实现

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. package com.gloomyfish.ii.demo;  
  2.   
  3. import java.awt.image.BufferedImage;  
  4.   
  5. /** 
  6.  * fast edge preserve filter algorithm base on integral image 
  7.  * @author zhigang jia 
  8.  * @E-mail:bfnh1998@hotmail.com 
  9.  * 
  10.  */  
  11. public class FastEPFilter extends AbstractImageOptionFilter {  
  12.     // 窗口半径大小  
  13.     private int xr;  
  14.     private int yr;  
  15.     private float sigma;  
  16.     public FastEPFilter() {  
  17.     }  
  18.       
  19.     public void setWinsize(int radius) {  
  20.         this.xr = radius;  
  21.         this.yr = radius;  
  22.     }  
  23.   
  24.     public float getSigma() {  
  25.         return sigma;  
  26.     }  
  27.   
  28.     public void setSigma(float sigma) {  
  29.         this.sigma = sigma;  
  30.     }  
  31.   
  32.     @Override  
  33.     public BufferedImage process(BufferedImage image) {  
  34.         long time = System.currentTimeMillis();  
  35.         int width = image.getWidth();  
  36.         int height = image.getHeight();  
  37.         // get image data  
  38.         int[] pixels = new int[width * height];  
  39.         int[] outPixels = new int[width * height];  
  40.         getRGB(image, 00, width, height, pixels);  
  41.         int size = (xr * 2 + 1) * (yr * 2 + 1);  
  42.         int r = 0, g = 0, b = 0;  
  43.         float sigma2 = sigma*sigma;  
  44.           
  45.         // per-calculate integral image  
  46.         byte[] R = new byte[width*height];  
  47.         byte[] G = new byte[width*height];  
  48.         byte[] B = new byte[width*height];  
  49.         getRGB(width, height, pixels, R, G, B);  
  50.         IntIntegralImage rii = new IntIntegralImage();  
  51.         rii.setImage(R);  
  52.         rii.process(width, height);  
  53.         IntIntegralImage gii = new IntIntegralImage();  
  54.         gii.setImage(G);  
  55.         gii.process(width, height);  
  56.         IntIntegralImage bii = new IntIntegralImage();  
  57.         bii.setImage(B);  
  58.         bii.process(width, height);  
  59.         int index = 0;  
  60.           
  61.         for (int row = yr; row < height - yr; row++) {  
  62.             for (int col = xr; col < width - xr; col++) {  
  63.                 index = row * width + col;  
  64.                 r = ((pixels[index] >> 16) & 0xff);  
  65.                 g = (pixels[index] >> 8) & 0xff;  
  66.                 b = (pixels[index] & 0xff);  
  67.                   
  68.                 int sr = rii.getBlockSum(col, row, (yr * 2 + 1), (xr * 2 + 1));  
  69.                 int sg = gii.getBlockSum(col, row, (yr * 2 + 1), (xr * 2 + 1));  
  70.                 int sb = bii.getBlockSum(col, row, (yr * 2 + 1), (xr * 2 + 1));  
  71.                   
  72.                 float vr = rii.getBlockSquareSum(col, row, (yr * 2 + 1), (xr * 2 + 1));  
  73.                 float vg = gii.getBlockSquareSum(col, row, (yr * 2 + 1), (xr * 2 + 1));  
  74.                 float vb = bii.getBlockSquareSum(col, row, (yr * 2 + 1), (xr * 2 + 1));  
  75.                   
  76.                 // 计算均值  
  77.                 float mr = sr / size;  
  78.                 float mg = sg / size;  
  79.                 float mb = sb / size;  
  80.                   
  81.                 // 计算方差  
  82.                 float dr = (vr - (sr*sr)/size)/size;  
  83.                 float dg = (vg - (sg*sg)/size)/size;  
  84.                 float db = (vb - (sb*sb)/size)/size;  
  85.                   
  86.                 // 计算系数K  
  87.                 float kr = dr / (dr+sigma2);  
  88.                 float kg = dg / (dg+sigma2);  
  89.                 float kb = db / (db+sigma2);  
  90.                   
  91.                 // 得到滤波后的像素值  
  92.                 r = (int)((1-kr)*mr + kr*r);  
  93.                 g = (int)((1-kg)*mg + kg*g);  
  94.                 b = (int)((1-kb)*mb + kb*b);  
  95.                   
  96.                 outPixels[row * width + col] = (0xff << 24) | (clamp(r) << 16) | (clamp(g) << 8) | clamp(b);  
  97.             }  
  98.         }  
  99.         System.out.println("FastEPFilter ->> time duration : " + (System.currentTimeMillis() - time));  
  100.         BufferedImage dest = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);  
  101.         setRGB(dest, 00, width, height, outPixels);  
  102.         return dest;  
  103.     }  
  104.       
  105.     /** Returns the red, green and blue planes as 3 byte arrays. */  
  106.     public void getRGB(int width, int height, int[] pixels, byte[] R, byte[] G, byte[] B) {  
  107.         int c, r, g, b;  
  108.         for (int i=0; i < width*height; i++) {  
  109.             c = pixels[i];  
  110.             r = (c&0xff0000)>>16;  
  111.             g = (c&0xff00)>>8;  
  112.             b = c&0xff;  
  113.             R[i] = (byte)r;  
  114.             G[i] = (byte)g;  
  115.             B[i] = (byte)b;  
  116.         }  
  117.     }  
  118.   
  119. }  

四:运行效果

半径设置为5,即窗口大小为5的时候,调整参数sigma的值即可得到此效果

其实很多磨皮的算法都是基于这个算法实现的,这个才是我想说的重点,

只有耐心看到此处才可以得到正确的答案。

五:参考:

http://imagej.NET/Integral_Image_Filters#Variance

http://www.activovision.com/octavi/doku.PHP?id=integral_images

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值