多线程03

SDWebImage框架详解

  • 下载图片并显示:
 [cell.imageView sd_setImageWithURL:[NSURL URLWithString:appM.icon] placeholderImage:[UIImage imageNamed:@"Snip20200808_172"]];
  • 下载图片/显示图片/内存缓存/磁盘缓存
   -(void)download1
{
 /*
  第一个参数:要下载图片的URL
  第二个参数:占位图片
  第三个参数:下载选项
  第四个参数:progress 进度回调
     receivedSize:已经下载的数据大小
     expectedSize:图片的中大小
  第五个参数:completed 完成回调(成功|失败)
     cacheType:是否使用了缓存,使用的方式
  */
 [self.imageView sd_setImageWithURL:[NSURL URLWithString:@"http://ww1.sinaimg.cn/crop.0.0.720.720.1024/abe7c97cjw8ermn0v2x7nj20k00k0jrz.jpg"] placeholderImage:[UIImage imageNamed:@"Snip20200808_11"] options:SDWebImageLowPriority | SDWebImageCacheMemoryOnly progress:^(NSInteger receivedSize, NSInteger expectedSize) {
     NSLog(@"%f",(CGFloat)receivedSize/expectedSize);
 } completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, NSURL *imageURL) {
 }];
}

  • 下载图片\内存缓存\磁盘缓存
  -(void)download2
{
  [[SDWebImageManager sharedManager]downloadImageWithURL:[NSURL URLWithString:@"http://img.kumi.cn/photo/6b/42/eb/6b42eb5597c4f174.jpg"] options:kNilOptions progress:^(NSInteger receivedSize, NSInteger expectedSize) {
        NSLog(@"%f",(CGFloat)receivedSize/expectedSize);
  } completed:^(UIImage *image, NSError *error, SDImageCacheType cacheType, BOOL finished, NSURL *imageURL) {
      if (error == nil) {
          self.imageView.image = image;
      }
  }];
}

  • 下载图片(完成后回调是在子线程中完成处理的)
-(void)download3
{
  [[SDWebImageDownloader sharedDownloader] downloadImageWithURL:[NSURL URLWithString:@"http://img.kumi.cn/photo/6b/42/eb/6b42eb5597c4f174.jpg"] options:kNilOptions progress:^(NSInteger receivedSize, NSInteger expectedSize) {

  } completed:^(UIImage *image, NSData *data, NSError *error, BOOL finished) {

      //completed是在子线程中处理的
      dispatch_async(dispatch_get_main_queue(), ^{
          //设置图片
          self.imageView.image = image;
      });

  }];
}
  • 显示gif动画
   self.imageView.image = [UIImage sd_animatedGIFNamed:@"1234"];
  • 接受到系统内存警告时如何处理:
      //(1)取消当前正在进行的所有下载操作
      [[SDWebImageManager sharedManager] cancelAll];
    
      //(2)清除缓存数据
      //cleanDisk:删除过期的文件数据,计算当前未过期的已经下载的文件数据的大小,如果发现该数据大小大于我们设置的最大缓存数据大小,那么程序内部会按照按文件数据缓存的时间从远到近删除,知道小于最大缓存数据为止。
      //clearMemory:直接删除文件,重新创建新的文件夹
      //[[SDWebImageManager sharedManager].imageCache cleanDisk];
      [[SDWebImageManager sharedManager].imageCache clearMemory];
    
  • SDWebImage内部实现细节:
  • 判断图片当前类型:之判断二进制数据的第一个字节
  • 默认缓存周期:一周
  • 缓存策略:默认进行内存和磁盘缓存,下载时首先检查内存缓存,其次是磁盘缓存
  • 缓存实现方式:采用了苹果推出的用来处理缓存的NSCache
  • 对内存警告的处理:框架内部监听系统内存警告的通知,当发生时,自动移除缓存中的所有对象
  • 下载队列中对多个图片任务采取的措施:方式有FIFO以及LIFO两种方式,默认是FIFO
  • 框架内允许的最大并发数6
  • 磁盘缓存图片的命名:对图片url进行md5散列加密(【echo -n "url" |MD5】)

NSCache详解

  • NSCache简单说明:
  • NSCache是苹果用来管理内存的类,类似于MutibleArray,在SDWebImage和AFN等框架中广泛用来管理缓存
  • NSCache在内存过低时会自动释放对象(我们要手动释放对象)
  • NSCache是线程安全的,在使用过程中不需要加锁
  • NSCache的Key只是对对象进行Strong引用,不是拷贝,在清理的时候计算的是实际大小而不是引用的大小(不明白)
  • NSCache属性以及方法介绍
  • 1)属性介绍
    • name:名称
    • delegete:设置代理
    • totalCostLimit:缓存空间的最大总成本,超出上限会自动回收对象。默认值为0,表示没有限制
    • countLimit:能够缓存的对象的最大数量。默认值为0,表示没有限制
    • evictsObjectsWithDiscardedContent:标识缓存是否回收废弃的内容
  • 2)方法介绍
    objc - (void)setObject:(ObjectType)obj forKey:(KeyType)key;//在缓存中设置指定键名对应的值,0成本 - (void)setObject:(ObjectType)obj forKey:(KeyType)keycost:(NSUInteger)g; //在缓存中设置指定键名对应的值,并且指定该键值对的成本,用于计算记录在缓存中的所有对象的总成本 //当出现内存警告或者超出缓存总成本上限的时候,缓存会开启一个回收过程,删除部分元素 - (void)removeObjectForKey:(KeyType)key;//删除缓存中指定键名的对象 - (void)removeAllObjects;//删除缓存中所有的对象

位移的简单说明

  • 常见的几种枚举形式:
//枚举一
  typedef enum{

      XMGDemoTypeTop,
      XMGDemoTypeBottom,

  }XMGDemoType;

  //枚举二
  typedef NS_ENUM(NSInteger,XMGType)
  {
      XMGTypeTop,
      XMGTypeBottom,
  };

  //枚举三:位移枚举
  typedef NS_OPTIONS(NSInteger, XMGActionType)
  {
      XMGActionTypeTop = 1<<0,
      XMGActionTypeBottom = 1<<1,
      XMGActionTypeLeft = 1<<2,
      XMGActionTypeRight = 1 <<3,
  };
  • 位移枚举相关说明
    • 特点:通过使用位移枚举可以实现一个参数实现传递多个操作
    • 原理:按位与只要有0则为0,按位或只要有1则为1
    • 技巧:如果位移枚举的第一个选项为0,那么在传递参数的时候默认可以传0,传0性能最优,不做额外的操作

RunLoop介绍:

  • 基础知识:

  • 基本作用:

  • 保证程序不退出(死循环)

  • 处理各种事件(触摸事件、定时器事件、selector事件等)

  • 节省cpu资源,提高程序性能,该运行时运行,该休息时休息

  • RunLoop对象

    • 在iOS开发中有两套api来访问Runloop
      foundation框架【NSRunloop】
      core foundation框架【CFRunloopRef】
    • NSRunLoop和CFRunLoopRef都代表着RunLoop对象,它们是等价的,可以互相转换
    • NSRunLoop是基于CFRunLoopRef的一层OC包装,所以要了解RunLoop内部结构,需要多研究CFRunLoopRef层面的API(Core Foundation层面)
  • RunLoop与线程关系

  • RunLoop与子线程关系:一个RunLoop对应唯一的一个线程

  • RunLoop生命周期:在第一次获取时创建,在相对应线程死亡的时候销毁

  • RunLoop的创建:主线程已经创建好,子线程需要手动创建

  • 如何获得RunLoop对象:

    • 获得当前RunLoop对象:
          //01 NSRunloop
            NSRunLoop * runloop1 = [NSRunLoop currentRunLoop];
          //02 CFRunLoopRef
            CFRunLoopRef runloop2 =   CFRunLoopGetCurrent();
      
    • 拿到当前对应程序的主线程
      //01 NSRunloop
        NSRunLoop * runloop1 = [NSRunLoop mainRunLoop];
      //02 CFRunLoopRef
        CFRunLoopRef runloop2 =   CFRunLoopGetMain();
    
    • 注意点:开一个子线程创建runloop,不是通过alloc init方法创建,而是直接通过调用currentRunLoop方法来创建,它本身是一个懒加载的。
    • 在子线程中,如果不主动获取Runloop的话,那么子线程内部是不会创建Runloop的。可以下载CFRunloopRef的源码,搜索_CFRunloopGet0,查看代码。
    • Runloop对象是利用字典来进行存储,而且key是对应的线程Value为该线程对应的Runloop。
    • RunLoop相关类
      • RunLoop运行原理图
2.png

- RunLoop与相关类之间的关系图

1.png
5)CFRunloopTimerRef
        (1)NSTimer相关代码
            /*
                说明:
                (1)runloop一启动就会选中一种模式,当选中了一种模式之后其它的模式就都不鸟。一个mode里面可以添加多个NSTimer,也就是说以后当创建NSTimer的时候,可以指定它是在什么模式下运行的。
                (2)它是基于时间的触发器,说直白点那就是时间到了我就触发一个事件,触发一个操作。基本上说的就是NSTimer
                (3)相关代码
            */
            - (void)timer2
            {
                //NSTimer 调用了scheduledTimer方法,那么会自动添加到当前的runloop里面去,而且runloop的运行模式kCFRunLoopDefaultMode

                NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];

                //更改模式
                [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

            }

            - (void)timer1
            {
                //    [NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];

                NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(run) userInfo:nil repeats:YES];

                //定时器添加到UITrackingRunLoopMode模式,一旦runloop切换模式,那么定时器就不工作
                //    [[NSRunLoop currentRunLoop] addTimer:timer forMode:UITrackingRunLoopMode];

                //定时器添加到NSDefaultRunLoopMode模式,一旦runloop切换模式,那么定时器就不工作
                //    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

                //占位模式:common modes标记
                //被标记为common modes的模式 kCFRunLoopDefaultMode  UITrackingRunLoopMode
                [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

                //    NSLog(@"%@",[NSRunLoop currentRunLoop]);
            }

            - (void)run
            {
                NSLog(@"---run---%@",[NSRunLoop currentRunLoop].currentMode);
            }

            - (IBAction)btnClick {

                NSLog(@"---btnClick---");
            }

        (2)GCD中的定时器
            //0.创建一个队列
            dispatch_queue_t queue = dispatch_get_global_queue(0, 0);

            //1.创建一个GCD的定时器
            /*
             第一个参数:说明这是一个定时器
             第四个参数:GCD的回调任务添加到那个队列中执行,如果是主队列则在主线程执行
             */
            dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);

            //2.设置定时器的开始时间,间隔时间以及精准度

            //设置开始时间,三秒钟之后调用
            dispatch_time_t start = dispatch_time(DISPATCH_TIME_NOW,3.0 *NSEC_PER_SEC);
            //设置定时器工作的间隔时间
            uint64_t intevel = 1.0 * NSEC_PER_SEC;

            /*
             第一个参数:要给哪个定时器设置
             第二个参数:定时器的开始时间DISPATCH_TIME_NOW表示从当前开始
             第三个参数:定时器调用方法的间隔时间
             第四个参数:定时器的精准度,如果传0则表示采用最精准的方式计算,如果传大于0的数值,则表示该定时切换i可以接收该值范围内的误差,通常传0
             该参数的意义:可以适当的提高程序的性能
             注意点:GCD定时器中的时间以纳秒为单位(面试)
             */

            dispatch_source_set_timer(timer, start, intevel, 0 * NSEC_PER_SEC);

            //3.设置定时器开启后回调的方法
            /*
             第一个参数:要给哪个定时器设置
             第二个参数:回调block
             */
            dispatch_source_set_event_handler(timer, ^{
                NSLog(@"------%@",[NSThread currentThread]);
            });

            //4.执行定时器
            dispatch_resume(timer);

            //注意:dispatch_source_t本质上是OC类,在这里是个局部变量,需要强引用
            self.timer = timer;

            GCD定时器补充
            /*
             DISPATCH_SOURCE_TYPE_TIMER         定时响应(定时器事件)
             DISPATCH_SOURCE_TYPE_SIGNAL        接收到UNIX信号时响应

             DISPATCH_SOURCE_TYPE_READ          IO操作,如对文件的操作、socket操作的读响应
             DISPATCH_SOURCE_TYPE_WRITE         IO操作,如对文件的操作、socket操作的写响应
             DISPATCH_SOURCE_TYPE_VNODE         文件状态监听,文件被删除、移动、重命名
             DISPATCH_SOURCE_TYPE_PROC          进程监听,如进程的退出、创建一个或更多的子线程、进程收到UNIX信号

             下面两个都属于Mach相关事件响应
                DISPATCH_SOURCE_TYPE_MACH_SEND
                DISPATCH_SOURCE_TYPE_MACH_RECV
             下面两个都属于自定义的事件,并且也是有自己来触发
                DISPATCH_SOURCE_TYPE_DATA_ADD
                DISPATCH_SOURCE_TYPE_DATA_OR
             */

6)CFRunloopSourceRef
        (1)是事件源也就是输入源,有两种分类模式;
              a.一种是按照苹果官方文档进行划分的
              b.另一种是基于函数的调用栈来进行划分的(source0和source1)。
        (2)具体的分类情况
            a.以前的分法
                Port-Based Sources
                Custom Input Sources
                Cocoa Perform Selector Sources
            b.现在的分法
                Source0:非基于Port的
                Source1:基于Port的
        (3)可以通过打断点的方式查看一个方法的函数调用栈

    7)CFRunLoopObserverRef
        (1)CFRunLoopObserverRef是观察者,能够监听RunLoop的状态改变
        (2)如何监听
             //创建一个runloop监听者
                CFRunLoopObserverRef observer = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(),kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {

                    NSLog(@"监听runloop状态改变---%zd",activity);
                });

                //为runloop添加一个监听者
                CFRunLoopAddObserver(CFRunLoopGetCurrent(), observer, kCFRunLoopDefaultMode);

                CFRelease(observer);
        (3)监听的状态
            typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) {
                kCFRunLoopEntry = (1UL << 0),   //即将进入Runloop
                kCFRunLoopBeforeTimers = (1UL << 1),    //即将处理NSTimer
                kCFRunLoopBeforeSources = (1UL << 2),   //即将处理Sources
                kCFRunLoopBeforeWaiting = (1UL << 5),   //即将进入休眠
                kCFRunLoopAfterWaiting = (1UL << 6),    //刚从休眠中唤醒
                kCFRunLoopExit = (1UL << 7),            //即将退出runloop
                kCFRunLoopAllActivities = 0x0FFFFFFFU   //所有状态改变
            };

3)Runloop运行逻辑


3.png
4.png

4.Runloop应用
1)NSTimer
2)ImageView显示:控制方法在特定的模式下可用
3)PerformSelector
4)常驻线程:在子线程中开启一个runloop
5)自动释放池
第一次创建:进入runloop的时候
最后一次释放:runloop退出的时候
其它创建和释放:当runloop即将休眠的时候会把之前的自动释放池释放,然后重新创建一个新的释放池

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,783评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,360评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,942评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,507评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,324评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,299评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,685评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,358评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,652评论 1 293
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,704评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,465评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,318评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,711评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,991评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,265评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,661评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,864评论 2 335

推荐阅读更多精彩内容