IOS Core Image之一

项目中要实现高斯模糊的效果,今天看了下Core Image这块的内容, 主要包括CIImage、CIFilter、CIContext、CIDetector(检测)、CIFeature(特征)等类。

今天先记录下CIImage、CIFilter、CIContext三个类的使用。

一、基本的滤镜效果需要以下步骤

1.创建CIImage对象

2.创建CIContext对象用作画布

3.创建CIFilter对象

4.输出滤镜

二、创建上面三个对象的API

1.创建CIImage对象 主要通过以下方法( 方法有好多种  具体查看类CIImage)

+ (CIImage *)imageWithCGImage:(CGImageRef)image;
+ (CIImage *)imageWithCGLayer:(CGLayerRef)layer;
+ (nullable CIImage *)imageWithContentsOfURL:(NSURL *)url;
+ (nullable CIImage *)imageWithData:(NSData *)data;
- (instancetype)initWithCGImage:(CGImageRef)image;
- (nullable instancetype)initWithData:(NSData *)data;
- (nullable instancetype)initWithContentsOfURL:(NSURL *)url;
- (instancetype)initWithColor:(CIColor *)color;

2.创建CIContext对象  

CIContext 构造函数contextWithOptions:的输入是一个NSDictionary。 它规定了各种选项,包括颜色格式以及内容是否应该运行在CPU或是GPU上。

 CIContext *context=[CIContext contextWithOptions:nil];

在该类中还有一些其他方法

- (void)drawImage:(CIImage *)image
          atPoint:(CGPoint)atPoint
         fromRect:(CGRect)fromRect NS_DEPRECATED(10_4,10_8, 5_0,6_0);

/* Render the rectangle 'fromRect' of 'image' to the rectangle 'inRect' in the
 * context's destination. */
- (void)drawImage:(CIImage *)image
           inRect:(CGRect)inRect
         fromRect:(CGRect)fromRect;

/* Render the region 'fromRect' of image 'image' into a temporary buffer using
 * the context, then create and return a new CoreGraphics image with
 * the results. The caller is responsible for releasing the returned
 * image. */
- (CGImageRef)createCGImage:(CIImage *)image
                   fromRect:(CGRect)fromRect
CF_RETURNS_RETAINED;

/* Create a new CGImage from the specified subrect of the image. If
 * non-nil the new image will be created in the specified format and
 * colorspace. */
- (CGImageRef)createCGImage:(CIImage *)image
                   fromRect:(CGRect)fromRect
                     format:(CIFormat)format
                 colorSpace:(nullable CGColorSpaceRef)colorSpace

3.创建CIFilter对象

1、创建滤镜对象 

+ (nullable CIFilter *) filterWithName:(NSString *) name;

/** Creates a new filter of type 'name'.
 The filter's input parameters are set from the list of key-value pairs which must be nil-terminated.
 On OSX, any of the filter input parameters not specified in the list will be undefined.
 On iOS, any of the filter input parameters not specified in the list will be set to default values. */
+ (nullable CIFilter *)filterWithName:(NSString *)name
                        keysAndValues:key0, ... NS_REQUIRES_NIL_TERMINATION NS_SWIFT_UNAVAILABLE("");

/** Creates a new filter of type 'name'.
 The filter's input parameters are set from the dictionary of key-value pairs.
 On OSX, any of the filter input parameters not specified in the dictionary will be undefined.
 On iOS, any of the filter input parameters not specified in the dictionary will be set to default values. */
+ (nullable CIFilter *)filterWithName:(NSString *)name
                  withInputParameters:(nullable CI_DICTIONARY(NSString*,id) *)params NS_AVAILABLE(10_10, 8_0);

2、上面创建滤镜对象时需要filterName,那怎么查看name以及每个CIFilter对象的属性呢?

CIFilter *filter = [CIFilter filterWithName:@"CIGaussianBlur"];
    NSLog(@"%@",filter.attributes);

在上面的代码中输出以下代码

2015-11-10 16:15:46.697 CoreImage[22672:246714] {
    "CIAttributeFilterAvailable_Mac" = "10.4";
    "CIAttributeFilterAvailable_iOS" = 6;
    CIAttributeFilterCategories =     (
        CICategoryBlur,
        CICategoryStillImage,
        CICategoryVideo,
        CICategoryBuiltIn
    );
    CIAttributeFilterDisplayName = "Gaussian Blur";
    CIAttributeFilterName = CIGaussianBlur;
    CIAttributeReferenceDocumentation = "http://developer.apple.com/cgi-bin/apple_ref.cgi?apple_ref=//apple_ref/doc/filter/ci/CIGaussianBlur";
    inputImage =     {
        CIAttributeClass = CIImage;
        CIAttributeDescription = "The image to use as an input image. For filters that also use a background image, this is the foreground image.";
        CIAttributeDisplayName = Image;
        CIAttributeType = CIAttributeTypeImage;
    };
    inputRadius =     {
        CIAttributeClass = NSNumber;
        CIAttributeDefault = 10;
        CIAttributeDescription = "The radius determines how many pixels are used to create the blur. The larger the radius, the blurrier the result.";
        CIAttributeDisplayName = Radius;
        CIAttributeIdentity = 0;
        CIAttributeMin = 0;
        CIAttributeSliderMax = 100;
        CIAttributeSliderMin = 0;
        CIAttributeType = CIAttributeTypeScalar;
    };
}

从上面的输出结果我们可以看到filter有CIAttributeFilterCategories、CIAttributeFilterDisplayName、CIAttributeFilterName、inputImage、inputRadius等属性。

在CIAttributeFilterCategories中可以看到滤镜有CICategoryBlur, CICategoryStillImage,CICategoryVideo, CICategoryBuiltIn种,在CIFilter类中有下面的两个方法能遍历出所有的滤镜名

/** Returns an array containing all published filter names in a category. */
+ (CI_ARRAY(NSString*) *)filterNamesInCategory:(nullable NSString *)category;

/** Returns an array containing all published filter names that belong to all listed categories. */
+ (CI_ARRAY(NSString*) *)filterNamesInCategories:(nullable CI_ARRAY(NSString*) *)categories;

获得到滤镜名之后就可以通过attributes属性查看filter对象的属性 ,通过KVC来设置属性值

//和UIButton类似 通过简单参数确定CIFilter子类  简单工厂
    CIFilter *filter = [CIFilter filterWithName:@"CIAffineTransform"];
    //  NSLog(@"%@",filter.attributes);
    //将图片输入到滤镜中,打个比方:滤镜是个具有某个功能的容器,任何东西放进去,拿出来的时候就会附上效果。 KVC
    [filter setValue:ciimage forKey:@"inputImage"];
    [filter setValue: [NSValue valueWithCGAffineTransform:CGAffineTransformMakeRotation(30)] forKey:@"inputTransform"];

4.输出滤镜

1、通过UIImage的imageWithCIImage方法

//从滤镜容器中取出图片  这里还有一种输出方式:使用CIcontext  不知道这两种有什么优缺点,欢迎留言
    //CIImage *new = [filter valueForKey:kCIOutputImageKey];
    CIImage *outputImage = filter.outputImage;

    UIImageView *img = [[UIImageView alloc] initWithFrame:self.view.frame];
    img.image = [UIImage imageWithCIImage:outputImage];
    //img.image=[UIImage imageWithCIImage:outputImage scale:30 orientation:UIImageOrientationDown];
    [self.view addSubview:img];

2、通过通过UIImage的imageWithCGImage方法 此方法要用到CIContext

//CIImage *new = [filter valueForKey:kCIOutputImageKey];
    CIImage *outputImage = filter.outputImage;

    CIContext *context=[CIContext contextWithOptions:nil];
    CGImageRef cgimg =[context createCGImage:outputImage fromRect:[outputImage extent]];

    UIImageView *img = [[UIImageView alloc] initWithFrame:self.view.frame];
    img.image=[UIImage imageWithCGImage:cgimg];
    [self.view addSubview:img];
    CGImageRelease(cgimg);

上面两种方法中,第一个每次调用都会生成一个CIContext。CIContext本来是可以重用以便提高性能和效率的。

5.完整代码

//
//  ViewController.m
//  CoreImage
//
//  Created by City--Online on 15/11/10.
//  Copyright  2015年 City--Online. All rights reserved.
//

#import "ViewController.h"
#import <CoreImage/CoreImage.h>

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    CIImage *inputImg = [[CIImage alloc]initWithCGImage:[UIImage imageNamed:@"1.jpg"].CGImage];

    //和UIButton类似 通过简单参数确定CIFilter子类  简单工厂
    CIFilter *filter = [CIFilter filterWithName:@"CIAffineTransform"];

    //输出属性
    NSLog(@"%@",filter.attributes);

    //将图片输入到滤镜中,打个比方:滤镜是个具有某个功能的容器,任何东西放进去,拿出来的时候就会附上效果。 KVC
    [filter setValue:inputImg forKey:@"inputImage"];
    [filter setValue: [NSValue valueWithCGAffineTransform:CGAffineTransformMakeRotation(30)] forKey:@"inputTransform"];

    //从滤镜容器中取出图片  这里还有一种输出方式:使用CIcontext  不知道这两种有什么优缺点,欢迎留言
    //CIImage *new = [filter valueForKey:kCIOutputImageKey];
    CIImage *outputImage = filter.outputImage;

    //通过CIContext上下文 、imageWithCGImage输出滤镜
    CIContext *context=[CIContext contextWithOptions:nil];
    CGImageRef cgimg =[context createCGImage:outputImage fromRect:[outputImage extent]];

    UIImageView *img = [[UIImageView alloc] initWithFrame:self.view.frame];
    img.image=[UIImage imageWithCGImage:cgimg];
    [self.view addSubview:img];
    CGImageRelease(cgimg);

    //遍历每种滤镜下的滤镜名
    NSLog(@"%@",[CIFilter filterNamesInCategories:@[@"CICategoryBlur"]]);
    NSLog(@"%@",[CIFilter filterNamesInCategories:@[@"CICategoryVideo"]]);
    NSLog(@"%@",[CIFilter filterNamesInCategories:@[@"CICategoryStillImage"]]);
    NSLog(@"%@",[CIFilter filterNamesInCategories:@[@"CICategoryBuiltIn"]]);

}

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

6.效果

时间: 2024-09-20 00:51:44

IOS Core Image之一的相关文章

IOS Core Image之二

在上篇博客IOS Core Image之一中了解了下CIImage.CIFilter.CIContext三个类的使用,这篇了解下滤镜链(多滤镜)和人脸检测(不是人脸识别). 一.多滤镜 1.有些效果不只是一个滤镜能完成的,需要多个滤镜叠加,让一个滤镜的outputImage作为另一个滤镜的inputImage. 在下面的代码中,给图片加了两个滤镜效果一个高斯模糊一个旋转. #import "ViewController.h" #import <CoreImage/CoreImag

iOS - Core Animation 核心动画

1.UIView 动画 具体讲解见 iOS - UIView 动画 2.UIImageView 动画 具体讲解见 iOS - UIImageView 动画 3.CADisplayLink 定时器 具体讲解见 iOS - OC NSTimer 定时器 CADisplayLink 是一个能让我们以和屏幕刷新率相同的频率将内容画到屏幕上的定时器.我们在应用中创建一个新的 CADisplayLink 对象,把它添加到一个 runloop 中,并给它提供一个 target 和 selector 在屏幕刷新

iOS Core Data的returnsObjectsAsFaults属性

来自论坛的一个问题:[CoreData] returnsObjectsAsFaults是什么用的. 这个属性是用来做优化的,但是有时候反而会降低性能,打个简单的比方,假如你有一个Department对象,它和Employee对象是一对多的关系(比如一个部门有100个员工),当你加载Department的时候,它包含的所有Employee也被加载了,此时如果returnsObjectsAsFaults为YES,则员工们不会被添加到内存中,而是被放在了row cache里,Department对象里

iOS - Core Animation 核心动画的使用

1.简单使用示例 1.1 时钟 QClockView.h @interface QClockView : UIView /// 创建时钟界面 + (instancetype)q_clockViewWithFrame:(CGRect)frame; @end QClockView.m #define CLOCK_WIDTH self.bounds.size.width @interface QClockView () // 表盘 @property (nonatomic, strong) UIIma

ios-IOS Core Data部分两个Entity之间的RelationShip的存储问题

问题描述 IOS Core Data部分两个Entity之间的RelationShip的存储问题 最近看部分IOS开发视频和书籍时有个问题没想明白: 1.就是在Data Model中创建两个Entity间的Relationship时,这个Relationship会**以某种形式存入数据库**吗? 2.如果存入数据库,那么是否表示当第一次创建Entity间的relationship后,以后每次一个Entity通过relationship获取其他Entity时系统会自动做类似表的连接操作? 3.如果

iOS面试一般性问题

原文出自:标哥的技术博客 前言 面试题中有一些一般性的问题,通常是会问到的.面试iOS应聘者时,切入点很重要,不同的切入点会导致不同的结果,没有找到合适的切入点也无法对应聘者有一个全面的了解.所以下面的面试问题更多的是提供方向,没有固定的答案,而且可以根据应聘者的回应引出更多有意思深层次的讨论. 注意:以下问题的参考答案均为笔者所答,不代表正确,问题答案因人而异,请根据自己的实际情况回答,若认为不合理,请在评论中指出.下面所有的参考答案,都是笔者站在面试官的角度来分析的,不同的面试官也会不一样.

iOS如何进行优化项目?

1. 结构与架构 1.1 结构 这里说的结构大概有两点:1.文件目录分类 2.第三方库管理 1.1.1 文件目录分类 为了方便管理,最好将Xcode中的项目展示目录与实际的存储目录保持一致 此外,一般按业务模块分类,一级目录可以按照MVC格式,也可以按照业务模块划分 用最普遍的Model View Controller架构举例 以一个基础的电商项目来解释,4个tabbarItem对应着四大模块,首页.分类.购物车.个人中心,往下每个还可以细分为MVC+Session层 按项目架构来分 最外层为M

蓝牙4.0 For IOS

蓝牙4.0 For IOS 用于IOS,更多说的是开发有关 目录  [隐藏]  1 更多研究 2 发现 3 调试用APP 4 困扰 5 NRF8002 6 一些研究 6.1 官方Demo 7 CC2540官方源码 8 经验 9 IOS的蓝牙4.0支持 10 IOS库支持 11 官方文档说法 12 见识 12.1 [iOS][RCtank]在xcode的iOS虚拟机中对BLE(蓝牙4.0)进行调试_jianglurong_新浪博客 沿途见识[2] 12.2 在xcode的ios模拟器里调试BLE_

ios-IOS Core Data 通过条件查询数据库

问题描述 IOS Core Data 通过条件查询数据库 数据库中已经存入数据,在使用的查询条件是request.predicate = NSPredicate(format: "id == %i", student.ID!) 数据库中的id使用的是integer32, student.ID的类型也是这个. 通过这个条件查不到任何记录.同样的我通过查询条件request.predicate = NSPredicate(format: "name == %@" ,st