iOS展示gif图

展示本地gif图片
SDWebImage比较占内存,FLAnimatedImage不怎么占用内存

1.使用SDWebImage

  1. 导入头文件
    #import "UIImage+GIF.h"
  2. 设置gif图
    self.imageView.image = [UIImage sd_animatedGIFNamed:@"test"];
    也可以通过NSData导入
    + (UIImage *)sd_animatedGIFWithData:(NSData *)data

2.使用FLAnimatedImage

  1. 导入头文件
    #import "FLAnimatedImage.h"
  2. 设置gif图
    NSString *path = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"gif"];
    NSData *data = [NSData dataWithContentsOfFile:path];
    FLAnimatedImage *image = [FLAnimatedImage animatedImageWithGIFData:data];
    self.imageView.animatedImage = image;

3.SDWebImage实现原理

使用UIImage + GIF分类

1.将gif图的每一帧导出为一个UIImage,并且算出每一帧的播放时间,将导出的image放入images数组中
2.使用+ animatedImageWithImages: duration: 方法创建image,参数为第一步的images数组和播放总时长
3.设置UIImageView的image时用sd_animatedGIFNamed:方法即可展示gif

方法1: 通过NSData获取gif的UIImage
方法名:+ (UIImage )sd_animatedGIFWithData:(NSData *)data;
  1. 获取gif数据,及其帧数
  2. 获取每帧的像素位图,求出每帧的播放时间
  3. 按照屏幕分辨率生成UIImage并放入数组
  4. 通过+ animatedImageWithImages: duration:生成UIImage
    + (UIImage *)sd_animatedGIFWithData:(NSData *)data {
    if (!data) {
    return nil;
    }
    //通过CFData读取gif文件的数据
    CGImageSourceRef source = CGImageSourceCreateWithData((__bridge CFDataRef)data, NULL);
    //获取gif文件的帧数
    size_t count = CGImageSourceGetCount(source);
    UIImage *animatedImage;
    if (count <= 1) {
    animatedImage = [[UIImage alloc] initWithData:data];
    }
    else {//大于一张图片时
    NSMutableArray *images = [NSMutableArray array];
    //设置gif播放的时间
    NSTimeInterval duration = 0.0f;
    for (size_t i = 0; i < count; i++) {
    //获取gif指定帧的像素位图
    CGImageRef image = CGImageSourceCreateImageAtIndex(source, i, NULL);
    if (!image) {
    continue;
    }
    //获取每张图的播放时间
    duration += [self sd_frameDurationAtIndex:i source:source];
    [images addObject:[UIImage imageWithCGImage:image scale:[UIScreen mainScreen].scale orientation:UIImageOrientationUp]];
    CGImageRelease(image);
    }
    if (!duration) {//如果播放时间为空
    duration = (1.0f / 10.0f) * count;
    }
    animatedImage = [UIImage animatedImageWithImages:images duration:duration];
    }
    CFRelease(source);
    return animatedImage;
    }
方法2:获取指定帧的image的播放时间
方法名:+ (float)sd_frameDurationAtIndex:(NSUInteger)index source:(CGImageSourceRef)source
  1. 获取指定帧图片的属性字典
  2. 获取图片的gif属性字典
  3. 获取每一帧的播放时间(如果小于0.1,设置为0.1)
    + (float)sd_frameDurationAtIndex:(NSUInteger)index source:(CGImageSourceRef)source {
    float frameDuration = 0.1f;
    //获取这一帧图片的属性字典
    CFDictionaryRef cfFrameProperties = CGImageSourceCopyPropertiesAtIndex(source, index, nil);
    NSDictionary *frameProperties = (__bridge NSDictionary *)cfFrameProperties;
    //获取gif属性字典
    NSDictionary *gifProperties = frameProperties[(NSString *)kCGImagePropertyGIFDictionary];
    //获取这一帧持续的时间
    NSNumber *delayTimeUnclampedProp = gifProperties[(NSString *)kCGImagePropertyGIFUnclampedDelayTime];
    if (delayTimeUnclampedProp) {
    frameDuration = [delayTimeUnclampedProp floatValue];
    }
    else {
    NSNumber *delayTimeProp = gifProperties[(NSString *)kCGImagePropertyGIFDelayTime];
    if (delayTimeProp) {
    frameDuration = [delayTimeProp floatValue];
    }
    }
    //如果帧数小于0.1,则指定为0.1
    if (frameDuration < 0.011f) {
    frameDuration = 0.100f;
    }
    CFRelease(cfFrameProperties);
    return frameDuration;
    }
方法3:通过图片的Name获取UIImage
方法名:+ (UIImage *)sd_animatedGIFNamed:(NSString *)name;
  1. 根据屏幕分辨率给图片自动加后缀(@2x)
  2. 从mainBundle中获取Data数据
  3. 调用方法 + sd_animatedGIFWithData:
    + (UIImage *)sd_animatedGIFNamed:(NSString *)name {
    //判断屏幕分辨率
    CGFloat scale = [UIScreen mainScreen].scale;
    if (scale > 1.0f) {
    NSString *retinaPath = [[NSBundle mainBundle] pathForResource:[name stringByAppendingString:@"@2x"] ofType:@"gif"];
    NSData *data = [NSData dataWithContentsOfFile:retinaPath];
    if (data) {
    return [UIImage sd_animatedGIFWithData:data];
    }
    NSString *path = [[NSBundle mainBundle] pathForResource:name ofType:@"gif"];
    data = [NSData dataWithContentsOfFile:path];
    if (data) {
    return [UIImage sd_animatedGIFWithData:data];
    }
    return [UIImage imageNamed:name];
    }
    else {
    NSString *path = [[NSBundle mainBundle] pathForResource:name ofType:@"gif"];
    NSData *data = [NSData dataWithContentsOfFile:path];
    if (data) {
    return [UIImage sd_animatedGIFWithData:data];
    }
    return [UIImage imageNamed:name];
    }
    }
方法4:对gif按比例进行缩放
方法名:- (UIImage *)sd_animatedImageByScalingAndCroppingToSize:(CGSize)size;
  1. 获取较大的缩放比例值,宽高等比缩放
  2. 调整位置,使缩放后的图居中
  3. 遍历self.images, 将图片缩放后导出放入数组
  4. 通过animatedImageWithImages:duration:返回image
    - (UIImage *)sd_animatedImageByScalingAndCroppingToSize:(CGSize)size {
    if (CGSizeEqualToSize(self.size, size) || CGSizeEqualToSize(size, CGSizeZero)) {
    return self;
    }
    CGSize scaledSize = size;
    CGPoint thumbnailPoint = CGPointZero;
    //获取较大的缩放比例值,宽高等比缩放
    CGFloat widthFactor = size.width / self.size.width;
    CGFloat heightFactor = size.height / self.size.height;
    CGFloat scaleFactor = (widthFactor > heightFactor) ? widthFactor : heightFactor;
    scaledSize.width = self.size.width * scaleFactor;
    scaledSize.height = self.size.height * scaleFactor;
    //调整位置,使缩放后的图居中
    if (widthFactor > heightFactor) {
    thumbnailPoint.y = (size.height - scaledSize.height) * 0.5;
    }
    else if (widthFactor < heightFactor) {
    thumbnailPoint.x = (size.width - scaledSize.width) * 0.5;
    }
    //遍历self.images, 将图片缩放后导出放入数组
    NSMutableArray *scaledImages = [NSMutableArray array];
    for (UIImage *image in self.images) {
    UIGraphicsBeginImageContextWithOptions(size, NO, 0.0);
    [image drawInRect:CGRectMake(thumbnailPoint.x, thumbnailPoint.y, scaledSize.width, scaledSize.height)];
    UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
    [scaledImages addObject:newImage];
    UIGraphicsEndImageContext();
    }
    return [UIImage animatedImageWithImages:scaledImages duration:self.duration];
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,393评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,790评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,391评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,703评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,613评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,003评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,507评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,158评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,300评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,256评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,274评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,984评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,569评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,662评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,899评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,268评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,840评论 2 339

推荐阅读更多精彩内容