学学人家的框架学无止境iOS源码探究相关

iOS-SDWebImage框架解析

2016-10-20  本文已影响222人  岁与禾

2016年8月2日

SDWebImage框架

1 框架的类图

imageimage

2 简单的用法

2.1 给UIImageView设置图片(有 内存缓存+磁盘缓存 )
//1 设置图片的URL
NSURL *url = [NSURL URLWithString:@"http://img4.duitang.com/uploads/blog/201310/18/20131018213446_smUw4.thumb.700_0.jpeg"];

//2 直接使用UIImageView+WebCache.h的分类
[self.imageView sd_setImageWithURL:url placeholderImage:[UIImage imageNamed:@"egopv_error_placeholder"] options:0 progress:^(NSInteger receivedSize, NSInteger expectedSize) {
    //下载进度
} completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, NSURL *imageURL) {
    switch (cacheType) {
        case SDImageCacheTypeNone:
            NSLog(@"缓存中没有--直接下载的");
            break;
            
        case SDImageCacheTypeDisk:
            NSLog(@"磁盘缓存--直接从磁盘中获取的,未再次下载");
            break;
            
        case SDImageCacheTypeMemory:
            NSLog(@"内存缓存--直接从内存中获取的,未再次下载");
            break;
    }
}];
2.2 直接下载图片(有 内存缓存+图片缓存)--SDWebImageManager
//1 获取图片下载地址
NSURL *url = [NSURL URLWithString:@"http://img4.duitang.com/uploads/blog/201310/18/20131018213446_smUw4.thumb.700_0.jpeg"];

//2 获取当前的管理器--单例
SDWebImageManager *manager = [SDWebImageManager sharedManager];

//3 下载图片
[manager downloadImageWithURL:url options:0 progress:^(NSInteger receivedSize, NSInteger expectedSize) {
    //下载进度
} completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, BOOL finished, NSURL *imageURL) {
    //当前线程是主线程,可以刷新UI
    self.imageView.image = image;
}];
2.3 直接下载图片(没有缓存处理)---SDWebImageDownloader

<b style="color:red">
注意:下载完毕后,是不会回到主线程的,需要我们自己回到主线程刷新UI
</b>

//1 获取图片下载地址
NSURL *url = [NSURL URLWithString:@"http://img4.duitang.com/uploads/blog/201310/18/20131018213446_smUw4.thumb.700_0.jpeg"];

//2 获取当前的下载器--单例
SDWebImageDownloader *downloader = [SDWebImageDownloader sharedDownloader];

//3 下载图片
[downloader downloadImageWithURL:url options:0 progress:^(NSInteger receivedSize, NSInteger expectedSize) {
    
} completed:^(UIImage *image, NSData *data, NSError *error, BOOL finished) {
    //注意,当前的线程是子线程,不能刷新UI,需要回到主线程刷新
    NSLog(@"%@",[NSThread currentThread]);
    //回主线程
    [[NSOperationQueue mainQueue]addOperationWithBlock:^{
        self.imageView.image = image;
    }];
}];
2.4 GIF图片的播放

内部原理:是讲gif图片读到内存,然后计算图片的帧数,然后将这些图片帧转换成图片后加入到图片数组中,通过UIImage的类方法animationImage来得到一个可动画的图片即可

//需要导入UIImage+Gif的分类
//直接导入gif图片
UIImage *image =  [UIImage sd_animatedGIFNamed:@"xhr"];
self.imageView.image = image;
2.5 外部内存警告处理--我们自己处理

当系统接收到内存警告的时候,我们需要将所有的任务取消并清空缓存。所以,我们需要在AppDelegate.m的applicationDidReceiveMemoryWarning的方法中进行取消任务、清空缓存的操作。

AppDelegate.m文件

-(void)applicationDidReceiveMemoryWarning:(UIApplication *)application
{
    //1 清空缓存
    [[[SDWebImageManager sharedManager] imageCache] clearMemory];

    //2 暂停任务
    [[SDWebImageManager sharedManager] cancelAll]
}

<b style="color:red">
注意:clean和clear的区别?(清空缓存的时候)

(1)clean,包含了cleanDisk、cleanMemory方法。他首先会将已经过期(默认过期时间为7天)的文件删除,然后将剩下的文件计算总大小,与预设的缓存最大字节进行比对,如果超过最大限制,会继续删除(删除规则是:按照时间先后顺序进行删除)。

(2)clear,包含clearDisk、clearMemory方法。直接全部清除。

</b>

3 框架内部结构详解

3.1 下载操作最大并发数(maxConcurrentOpeationCount)

开下载线程,一般会在类SDWebImageDownloader和SDWebImageDownloaderOperation中进行,所以,查看最大并发数也需要在这两个类中查找。

SDWebImageDownloader.m

-(id)init
{
    //其他代码省略....
    //设置最大并发数,默认是6条
    _downloadQueue.maxConcurrentOperationCount = 6;
    //其他代码省略....
}
3.2 下载队列中,默认的执行方式(executionOrder,执行顺序)

同上,仍然是在SDWebImageDownloader中

SDWebImageDownloader.m

-(id)init
{
    //设置执行的顺序
    _executionOrder = SDWebImageDownloaderFIFOExecutionOrder;
}
3.3 缓存文件的文件名

框架在缓存的时候,为了存储时尽可能隐藏图片信息,它的文件名是图片的URL进行MD5加密得到的。

<b style="color:orange">
tips: MacOS终端可以集成了MD5加密功能。输入命令:echo "待加密字符串" | md5

chenhuadeMacBook-Pro:~ chmn$ echo "hello world" | md5
</b>

3.4 框架内部对于系统内存告警的处理

框架内存也会接受系统的内存告警事件,并进行响应的操作。可以在SDImageCache中看到。它的实现思路是:通过注册系统内存告警的通知进行的。

[[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(clearMemory)
                                                 name:UIApplicationDidReceiveMemoryWarningNotification
                                               object:nil];

    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(cleanDisk)
                                                 name:UIApplicationWillTerminateNotification
                                               object:nil];

    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(backgroundCleanDisk)
                                                 name:UIApplicationDidEnterBackgroundNotification
                                               object:nil];
3.5 缓存的处理方式

我们自己写的简单多图下载程序中,缓存都是使用的可变字典,将图片的url和image进行对应。但是在框架中,它使用的NSCache(其实也类似字典)。

3.6 如何判断图片的类型

框架在下载文件,在保存文件的时候,需要知道文件的类型(jpg/jpeg/png/gif...)。因为同类型的图片转换成二进制(或十六进制)后,第一个字节是相同的,所以,框架是通过第一个字节去判断的。

NSData+ImageContentType.m

+ (NSString *)sd_contentTypeForImageData:(NSData *)data {
uint8_t c;
[data getBytes:&c length:1];
switch (c) {
    case 0xFF:
        return @"image/jpeg";
    case 0x89:
        return @"image/png";
    case 0x47:
        return @"image/gif";
    case 0x49:
    case 0x4D:
        return @"image/tiff";
    case 0x52:
        // R as RIFF for WEBP
        if ([data length] < 12) {
            return nil;
        }

        NSString *testString = [[NSString alloc] initWithData:[data subdataWithRange:NSMakeRange(0, 12)] encoding:NSASCIIStringEncoding];
        if ([testString hasPrefix:@"RIFF"] && [testString hasSuffix:@"WEBP"]) {
            return @"image/webp";
        }

        return nil;
}
return nil;
}
3.7 框架内部是如何下载图片的

是发送网络请求,之前的版本是NSURLConnnection,目前是NSURLSession

3.8 默认的请求超时时间

可以在SDWebImageDownloader.m文件查看,默认是的超时时间是15秒

SDWebImageDownloader.m

-(id)init
{
    _downloadTimeout = 15.0;
}
3.9 缓存的默认有效期

默认的缓存有效期是7天,可以在SDImageCache类中找到

static const NSInteger kDefaultCacheMaxCacheAge = 60 * 60 * 24 * 7; // 1 week
上一篇下一篇

猜你喜欢

热点阅读