iOS图片高斯模糊 毛玻璃效果

最近因为业务上面需要将图片进行高斯模糊,所以对这方面进行了一点简单的了解,在这里写出来,希望可以帮助到需要写这块的同学们。
图片模糊主要分为俩大类,一个是实用系统原生的类库来实现(Core Image、Accelerate-vImage),一个是实用第三方库(GPUImage)来实现,下面就进入正题

Core Image:在iOS 5.0之后就加入了这个API,在OS X和iOS这俩个平台上都可以使用这个类库,有大量的滤镜,直接上代码

CIContext *context = [CIContext contextWithOptions:nil];
CIImage *image = [CIImage imageWithContentsOfURL:imageURL];
CIFilter *filter = [CIFilter filterWithName:@"CIGaussianBlur"];
[filter setValue:image forKey:kCIInputImageKey];
[filter setValue:@2.0f forKey: @"inputRadius"];
CIImage *result = [filter valueForKey:kCIOutputImageKey];
CGImageRef outImage = [context createCGImage: result fromRect:[result extent]];
UIImage * blurImage = [UIImage imageWithCGImage:outImage];

这里只贴一段高斯模糊的代码,至于其它滤镜效果,可以点击链接查看CIImage API

2.Accelerate-vImage:使用时需要导入Accelerate类库,这个类库主要时用来做数字信号处理、图像处理相关的向量、矩阵运算的库。直接上代码

#pragma mark -高斯模糊图片处理
- (UIImage *)blurryImage:(UIImage *)image withBlurLevel:(CGFloat)blur {
    //模糊度
    if ((blur < 0.1f) || (blur > 2.0f)) {
        blur = 0.5f;
    }

    //boxSize必须大于0
    int boxSize = (int)(blur * 100);
    boxSize -= (boxSize % 2) + 1;
    //图像处理
    CGImageRef img = image.CGImage;

    //图像缓存,输入缓存,输出缓存
    vImage_Buffer inBuffer, outBuffer;
    vImage_Error error;
    //像素缓存
    void *pixelBuffer;

    //数据源提供者,Defines an opaque type that supplies Quartz with data.
    CGDataProviderRef inProvider = CGImageGetDataProvider(img);
    // provider’s data.
    CFDataRef inBitmapData = CGDataProviderCopyData(inProvider);

    //宽,高,字节/行,data
    inBuffer.width = CGImageGetWidth(img);
    inBuffer.height = CGImageGetHeight(img);
    inBuffer.rowBytes = CGImageGetBytesPerRow(img);
    inBuffer.data = (void*)CFDataGetBytePtr(inBitmapData);

    //像数缓存,字节行*图片高
    pixelBuffer = malloc(CGImageGetBytesPerRow(img) * CGImageGetHeight(img));

    outBuffer.data = pixelBuffer;
    outBuffer.width = CGImageGetWidth(img);
    outBuffer.height = CGImageGetHeight(img);
    outBuffer.rowBytes = CGImageGetBytesPerRow(img);


    // 第三个中间的缓存区,抗锯齿的效果
    void *pixelBuffer2 = malloc(CGImageGetBytesPerRow(img) * CGImageGetHeight(img));
    vImage_Buffer outBuffer2;
    outBuffer2.data = pixelBuffer2;
    outBuffer2.width = CGImageGetWidth(img);
    outBuffer2.height = CGImageGetHeight(img);
    outBuffer2.rowBytes = CGImageGetBytesPerRow(img);

    //Convolves a region of interest within an ARGB8888 source image by an implicit M x N kernel that has the effect of a box filter.
    error = vImageBoxConvolve_ARGB8888(&inBuffer, &outBuffer2, NULL, 0, 0, boxSize, boxSize, NULL, kvImageEdgeExtend);
    error = vImageBoxConvolve_ARGB8888(&outBuffer2, &inBuffer, NULL, 0, 0, boxSize, boxSize, NULL, kvImageEdgeExtend);
    error = vImageBoxConvolve_ARGB8888(&inBuffer, &outBuffer, NULL, 0, 0, boxSize, boxSize, NULL, kvImageEdgeExtend);


    if (error) {
        RNLog(@"error from convolution %ld", error);
    }

    // NSLog(@"字节组成部分:%zu",CGImageGetBitsPerComponent(img));
    //颜色空间DeviceRGB
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    //用图片创建上下文,CGImageGetBitsPerComponent(img),7,8
    CGContextRef ctx = CGBitmapContextCreate( outBuffer.data, outBuffer.width, outBuffer.height, 8, outBuffer.rowBytes, colorSpace, CGImageGetBitmapInfo(image.CGImage));

    //根据上下文,处理过的图片,重新组件
    CGImageRef imageRef = CGBitmapContextCreateImage (ctx);
    UIImage *returnImage = [UIImage imageWithCGImage:imageRef];

    //clean up
    CGContextRelease(ctx);
    CGColorSpaceRelease(colorSpace);

    free(pixelBuffer);
    free(pixelBuffer2);
    CFRelease(inBitmapData);
    CGImageRelease(imageRef);

    return returnImage;
}

3.GPUImage:十分强大的一个图像视频处理的第三方类库,赶快拉取代码开脑洞去吧github-GPUImage
只是使用第三方库的缺点就是,你需要导入大量的类库及第三方代码,如果只是使用模糊功能,会让你的包容量变大。直接上代码

GPUImageGaussianBlurFilter * blurFilter = [[GPUImageGaussianBlurFilter alloc] init];
blurFilter.blurRadiusInPixels = 2.0;
UIImage * image = [UIImage imageNamed:@"xxx"];
UIImage *blurredImage = [blurFilter imageByFilteringImage:image];

以下说一下使用这些方法需要注意的坑,如果是本地图片的话还没有发现大坑,如果是网络图片的话,最好要对这几个类库进行封装,因为会出现无图片链接或者有链接没有图片的情况,这样在进行core release的时候,会因为对空地址进行release而奔溃。
除了以上的坑,还发现一个有趣的小坑,就是[CIImage imageWithContentsOfURL:]的时候,如果是空白url的时候,在iOS8.0是可以正常运行,而在iOS9.0以后,就会奔溃。
以上就是高斯模糊的简单使用,大家可以留言进行交流。

你可能感兴趣的:(ios,iOS毛玻璃-iOS,iOS高斯模糊,iOS图片处理,iOS滤镜)