Inspired by http://incubator.quasimondo.com/processing/fastblur.pde , I wrote a piece of CSharp code to implement this efficient "gaussian blur" alogorithm for one of my .Net project.
The concept of the "fast" is avoiding floating point calculation and pre-calculating some multiplication operation, storing the results into a large two-dimension array for later blurring operations, so it saves lots of time.

Code:
using System;
using System.Collections.Generic;
using System.Text;
using System.Drawing;
using System.ComponentModel;

namespace Adrian.PhotoX.Lib
{
    [Serializable]
    
public enum BlurType
    {
        Both,
        HorizontalOnly,
        VerticalOnly,
    }

    [Serializable]
    [StyleTypeName(
"模糊")]
    
public class GaussianBlur : StyleBase
    {
        
private int _radius = 6;
        
private int[] _kernel;
        
private int _kernelSum;
        
private int[,] _multable;
        
private BlurType _blurType;

        
public GaussianBlur()
        {
            PreCalculateSomeStuff();
        }

        
public GaussianBlur(int radius)
        {
            _radius 
= radius;
            PreCalculateSomeStuff();
        }

        
private void PreCalculateSomeStuff()
        {
            
int sz = _radius * 2 + 1;
            _kernel 
= new int[sz];
            _multable 
= new int[sz, 256];
            
for (int i = 1; i <= _radius; i++)
            {
                
int szi = _radius - i;
                
int szj = _radius + i;
                _kernel[szj] 
= _kernel[szi] = (szi + 1* (szi + 1);
                _kernelSum 
+= (_kernel[szj] + _kernel[szi]);
                
for (int j = 0; j < 256; j++)
                {
                    _multable[szj, j] 
= _multable[szi, j] = _kernel[szj] * j;
                }
            }
            _kernel[_radius] 
= (_radius + 1* (_radius + 1);
            _kernelSum 
+= _kernel[_radius];
            
for (int j = 0; j < 256; j++)
            {
                _multable[_radius, j] 
= _kernel[_radius] * j;
            }
        }

        
public override Image ProcessImage(Image inputImage)
        {
            Bitmap origin 
= new Bitmap(inputImage);
            Bitmap blurred 
= new Bitmap(inputImage.Width, inputImage.Height);

            
using (RawBitmap src = new RawBitmap(origin))
            {
                
using (RawBitmap dest = new RawBitmap(blurred))
                {
                    
int pixelCount = src.Width * src.Height;
                    
int[] b = new int[pixelCount];
                    
int[] g = new int[pixelCount];
                    
int[] r = new int[pixelCount];

                    
int[] b2 = new int[pixelCount];
                    
int[] g2 = new int[pixelCount];
                    
int[] r2 = new int[pixelCount];

                    
int offset = src.GetOffset();
                    
int index = 0;
                    
unsafe
                    {
                        
byte* ptr = src.Begin;
                        
for (int i = 0; i < src.Height; i++)
                        {
                            
for (int j = 0; j < src.Width; j++)
                            {
                                b[index] 
= *ptr;
                                ptr
++;
                                g[index] 
= *ptr;
                                ptr
++;
                                r[index] 
= *ptr;
                                ptr
++;

                                
++index;
                            }
                            ptr 
+= offset;
                        }

                        
int bsum;
                        
int gsum;
                        
int rsum;
                        
int sum;
                        
int read;
                        
int start = 0;
                        index 
= 0;
                        
if (_blurType != BlurType.VerticalOnly)
                        {
                            
for (int i = 0; i < src.Height; i++)
                            {
                                
for (int j = 0; j < src.Width; j++)
                                {
                                    bsum 
= gsum = rsum = sum = 0;
                                    read 
= index - _radius;

                                    
for (int z = 0; z < _kernel.Length; z++)
                                    {
                                        
if (read >= start && read < start + src.Width)
                                        {
                                            bsum 
+= _multable[z, b[read]];
                                            gsum 
+= _multable[z, g[read]];
                                            rsum 
+= _multable[z, r[read]];
                                            sum 
+= _kernel[z];
                                        }
                                        
++read;
                                    }

                                    b2[index] 
= (bsum / sum);
                                    g2[index] 
= (gsum / sum);
                                    r2[index] 
= (rsum / sum);

                                    
if (_blurType == BlurType.HorizontalOnly)
                                    {
                                        
byte* pcell = dest[j, i];
                                        pcell[
0= (byte)(bsum / sum);
                                        pcell[
1= (byte)(gsum / sum);
                                        pcell[
2= (byte)(rsum / sum);
                                    }

                                    
++index;
                                }
                                start 
+= src.Width;
                            }
                        }
                        
if (_blurType == BlurType.HorizontalOnly)
                        {
                            
return blurred;
                        }

                        
int tempy;
                        
for (int i = 0; i < src.Height; i++)
                        {
                            
int y = i - _radius;
                            start 
= y * src.Width;
                            
for (int j = 0; j < src.Width; j++)
                            {
                                bsum 
= gsum = rsum = sum = 0;
                                read 
= start + j;
                                tempy 
= y;
                                
for (int z = 0; z < _kernel.Length; z++)
                                {
                                    
if (tempy >= 0 && tempy < src.Height)
                                    {
                                        
if (_blurType == BlurType.VerticalOnly)
                                        {
                                            bsum 
+= _multable[z, b[read]];
                                            gsum 
+= _multable[z, g[read]];
                                            rsum 
+= _multable[z, r[read]];
                                        }
                                        
else
                                        {
                                            bsum 
+= _multable[z, b2[read]];
                                            gsum 
+= _multable[z, g2[read]];
                                            rsum 
+= _multable[z, r2[read]];
                                        }
                                        sum 
+= _kernel[z];
                                    }
                                    read 
+= src.Width;
                                    
++tempy;
                                }

                                
byte* pcell = dest[j, i];
                                pcell[
0= (byte)(bsum / sum);
                                pcell[
1= (byte)(gsum / sum);
                                pcell[
2= (byte)(rsum / sum);
                            }
                        }
                    }
                }
            }

            
return blurred;
        }

        
public int Radius
        {
            
get { return _radius; }
            
set
            {
                
if (value < 2)
                {
                    
throw new InvalidOperationException("Radius must be greater then 1");
                }
                _radius 
= value;
                PreCalculateSomeStuff();
                RaisePropertyChanged(
thisnew PropertyChangedEventArgs("Radius"));
            }
        }

        
public BlurType BlurType
        {
            
get { return _blurType; }
            
set
            {
                _blurType 
= value;
                RaisePropertyChanged(
thisnew PropertyChangedEventArgs("BlurType"));
            }
        }
    }
}
P.S. RawBitmap is actually a wrapper class for BitmapData and Bitmap.LockBits operation
Source code download

Posted on 2007-03-30 22:17  Adrian H.  阅读(27344)  评论(6编辑  收藏  举报