我的编程空间,编程开发者的网络收藏夹
学习永远不晚

iOSNSCache和NSUrlCache缓存类实现示例详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

iOSNSCache和NSUrlCache缓存类实现示例详解

NSCache

NSCache是Foundation框架提供的缓存类的实现,使用方式类似于可变字典,最重要的是它是线程安全的,而NSMutableDictionary不是线程安全的,在多线程环境下使用NSCache是更好的选择。 类的基本属性和方法:

#import <Foundation/NSObject.h>
@class NSString;
@protocol NSCacheDelegate;
NS_ASSUME_NONNULL_BEGIN
API_AVAILABLE(macos(10.6), ios(4.0), watchos(2.0), tvos(9.0))
@interface NSCache <KeyType, ObjectType> : NSObject {
@private
    id _delegate;
    void *_private[5];
    void *_reserved;
}
@property (copy) NSString *name;
@property (nullable, assign) id<NSCacheDelegate> delegate;
- (nullable ObjectType)objectForKey:(KeyType)key;
- (void)setObject:(ObjectType)obj forKey:(KeyType)key; // 0 cost
- (void)setObject:(ObjectType)obj forKey:(KeyType)key cost:(NSUInteger)g;
- (void)removeObjectForKey:(KeyType)key;
- (void)removeAllObjects;
@property NSUInteger totalCostLimit;	// limits are imprecise/not strict
@property NSUInteger countLimit;	// limits are imprecise/not strict
@property BOOL evictsObjectsWithDiscardedContent;
@end
@protocol NSCacheDelegate <NSObject>
@optional
- (void)cache:(NSCache *)cache willEvictObject:(id)obj;
@end
NS_ASSUME_NONNULL_END

缓存淘汰策略

通过 GNUstep 提供的源码,我们得知其对于 NSCache 的处理是计算出一个平均访问次数,然后释放的是访问次数较少的对象,直到满足需要释放大小,也就是 LRU 的机制。通过 swift-corelibs-foundation 源码我们得知,其首先存储链表结构中是按对象花费内存大小排序的,然后通过用户有无指定 totalCostLimit 大小限制来依次释放(先释放占用较小的对象),直到满足需要释放大小。然后再通过个数限制来释放,直到满足需要释放大小(依旧是先释放较小的对象)。

GNUstepFoundation 源码地址:github.com/gnustep/lib…

Swift Foundation 源码地址:github.com/apple/swift…

在内存不足时NSCache会自动释放存储的对象。

NSCache的键key不会被复制,所以key不需要实现NSCopying协议。

可以设置缓存的最大数量,当缓存数量满的时候,再添加将先删除先添加的对象再增加。

唯一一个代理方法是一个对象将被删除时调用,调用方式有以下几种:

  • NSCache缓存对象自身被释放
  • 手动调用removeObjectForKey:方法
  • 手动调用removeAllObjects    
  • 缓存中对象的个数大于countLimit,或,缓存中对象的总cost值大于totalCostLimit    
  • 程序进入后台后
  • 收到系统的内存警告

基本用法:

@interface NSCacheViewController ()<NSCacheDelegate>
@property(nonatomic,strong) NSCache *cache;
@end
@implementation NSCacheViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    self.cache = [[NSCache alloc] init];
    //设置最大缓存数
    self.cache.countLimit = 5;
    //设置代理,实现代理方法,
    self.cache.delegate = self;
    self.cache.name = @"测试内存";
    for (int i=0; i<10; i++) {
        NSString *str = [NSString stringWithFormat:@"%d",i];
        [self.cache setObject:str forKey:str];
    }
    // Do any additional setup after loading the view.
}
//超出缓存部分会被释放, 收到内存警告时候系统也会释放一部分。
-(void)cache:(NSCache *)cache willEvictObject:(id)obj{
    NSLog(@"%@---%@",cache,obj);
}
@end

实际应用 SDWebImage

SDImageCacheConfig中可以配置是否使用内存做缓存,默认为YES

磁盘缓存的最大时长,默认为一周

SDImage中内存缓存SDMemoryCache继承与NScache,缓存时候会在NSCache和SDMemoryCache的NSMapTable各存一份。读取时候也优先读取系统cache,如果不存在再读取SDMemoryCache,这样做的目标是防止一些系统缓存不可控因素。

// `setObject:forKey:` just call this with 0 cost. Override this is enough
- (void)setObject:(id)obj forKey:(id)key cost:(NSUInteger)g {
    [super setObject:obj forKey:key cost:g];
    if (!self.config.shouldUseWeakMemoryCache) {
        return;
    }
    if (key && obj) {
        // Store weak cache
        SD_LOCK(_weakCacheLock);
        [self.weakCache setObject:obj forKey:key];
        SD_UNLOCK(_weakCacheLock);
    }
}

NSURLCache

使用缓存的目的是为了使应用程序能更快速的响应用户输入,是程序高效的运行。有时候我们需要将远程web服务器获取的数据缓存起来,以空间换取时间,减少对同一个url多次请求,减轻服务器的压力,优化客户端网络,让用户体验更良好。

背景:NSURLCache : 在iOS5以前,apple不支持磁盘缓存,在iOS5的时候,允许磁盘缓存,(NSURLCache 是根据NSURLRequest 来实现的)只支持http,在iOS6以后,支持http和https。

缓存的实现说明:由于GET请求一般用来查询数据,POST请求一般是发大量数据给服务器处理(变动性比较大),因此一般只对GET请求进行缓存,而不对POST请求进行缓存。

缓存原理:一个NSURLRequest对应一个NSCachedURLResponse

Etag全称是Entity Tag,一般用于标识URL对象是否发生了改变。 使用Etag一般会出现如下的请求流程:

Etag有点类似于文件hash或者说是信息摘要。

当进行一次URL请求,服务端会返回'Etag'响应头,下次浏览器请求相同的URL时,浏览器会自动将它设置为请求头'If-None-Match'的值。服务器收到这个请求之后,就开始做信息校验工作将自己本次产生的Etag与请求传递过来的'If-None-Match'对比,如果相同,则返回HTTP状态码304,并且response数据体中没有数据。

第二次请求的时候从哪里获取到'Etag'的值并赋给请求头'If-None-Match'的?自然是浏览器的缓存中取出的。那么浏览器收到304状态码之后又干了什么?刚才说到response数据体中没有数据,但是浏览器仍需加载页面,它会从缓存中读取上次缓存的页面。

//
//  NSURLCacheViewController.m
//  DemoTest2022
//
//  Created by wy on 2022/10/19.
//
#import "NSURLCacheViewController.h"
@interface NSURLCacheViewController ()
//去服务器比对资源是否需要更新
@property (nonatomic, strong) NSString *lastModified;
@property (nonatomic, strong) NSString *etag;
@property(nonatomic,strong) UIImageView *imagev;
@end
@implementation NSURLCacheViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    self.imagev = [[UIImageView alloc] initWithFrame:CGRectMake(0, 100, 100, 100)];
    [self.view addSubview:self.imagev];
    // Do any additional setup after loading the view.
}
-(void)requestTest{
    NSURL *url = [NSURL URLWithString:@"http://via.placeholder.com/50x50.png"];
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url cachePolicy:(NSURLRequestReloadIgnoringCacheData) timeoutInterval:15] ;
    if (self.lastModified) {
        [request setValue:self.lastModified forHTTPHeaderField:@"If-Modified-Since"];
    }
    if (self.etag) {
        [request setValue:self.etag forHTTPHeaderField:@"If-None-Match"];
    }
    [[[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
            if (error) {
                NSLog(@"error---%@",error);
            }else{
                NSData *tempData = data;
                NSString *responseStr = [[NSString alloc] initWithData:tempData encoding:NSUTF8StringEncoding];
                self.lastModified = [(NSHTTPURLResponse *)response  allHeaderFields][@"Last-Modified"];
                self.etag =[(NSHTTPURLResponse *)response allHeaderFields][@"etag"];
                NSLog(@"response:%@", response);
                dispatch_sync(dispatch_get_main_queue(), ^{
                    self.imagev.image = [UIImage imageWithData:tempData];
                });
            }
        }] resume] ;
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    [self requestTest];
}
@end

这样每次请求都使用忽略缓存的策略,但是要附带着"If-None-Match"头,它的值是上次请求的响应头"Etag"的值,于是服务器会每次都实时检查文件的修改状态,得到一个准确的状态值,最后决定返回304还是200。若是200,iOS则直接使用新的response和新的数据;如果是304,则使用新的response和缓存中的data。

这样既能够获取到最新的数据有能够节约带宽。两全其美。

iOS中定以的URLRequest缓存策略有以下几种:

typedef NS_ENUM(NSUInteger, NSURLRequestCachePolicy)
{
    NSURLRequestUseProtocolCachePolicy = 0,
    // 从不读取缓存,但请求后将response缓存起来
    NSURLRequestReloadIgnoringLocalCacheData = 1,
    NSURLRequestReloadIgnoringLocalAndRemoteCacheData = 4,
    NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData,
    // 以下两种在取缓存时,可能取到的是过期数据
    NSURLRequestReturnCacheDataElseLoad = 2,// 缓存中没有才去发起请求加载,有就不进行网络请求了
    NSURLRequestReturnCacheDataDontLoad = 3,// 缓存中没有不加载,绝不发起网络请求,缓存中没有则返回错误
    NSURLRequestReloadRevalidatingCacheData = 5,//Unimplemented
};

官方文档解释:

NSURLCache类通过将NSURLRequest对象映射到NSCachedURLResponse对象来实现URL加载请求响应的缓存。它提供了一个复合的内存和磁盘缓存,并允许您操作内存和磁盘部分的大小。您还可以控制缓存数据持久存储的路径。

在iOS中,当系统磁盘空间不足时,磁盘上的缓存可能会被清除,但只有在应用程序未运行时才会清除。

AFNetwork中用法:

总结:

NSCache 特点

  • 使用方便,类似字典
  • l线程安全
  • l内存不足时,NSCache会自动释放存储对象
  • NSCache的key不会被拷贝,不需要实现NSCopying协议
  • lNSDiscardableContent协议

NSURLCache主要应用与网络请求数据缓存策略,优化网络请求性能优化。

以上就是iOS NSCache和NSUrlCache缓存类实现示例详解的详细内容,更多关于iOS NSCache NSUrlCache的资料请关注编程网其它相关文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

iOSNSCache和NSUrlCache缓存类实现示例详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

iOSNSCache和NSUrlCache缓存类实现示例详解

这篇文章主要为大家介绍了iOSNSCache和NSUrlCache缓存类实现示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2022-11-13

Java实现LRU缓存的实例详解

Java实现LRU缓存的实例详解1.CacheCache对于代码系统的加速与优化具有极大的作用,对于码农来说是一个很熟悉的概念。可以说,你在内存中new 了一个一段空间(比方说数组,list)存放一些冗余的结果数据,并利用这些数据完成了以空
2023-05-31

MyBatis 动态SQL和缓存机制实例详解

有的时候需要根据要查询的参数动态的拼接SQL语句常用标签:- if:字符判断- choose【when...otherwise】:分支选择- trim【where,set】:字符串截取,其中where标签封装查询条件,set标签封装修改条件
2023-05-31

android 实现类似微信缓存和即时更新好友头像示例

引言使用微信时我们会发现,首次进入微信的好友列表时,会加载好友头像,但是再次进入时,就不用重新加载了,而且其他页面都不用重新加载,说明微信的好友头像是缓存在本地的,然后好友修改头像后,又会及时的更新,这个功能是如何实现的呢,我们来分析一下
2022-06-06

C++实现日期类的示例详解

这篇文章主要为大家详细介绍了四个C++常用的日期类的实现,文中的示例代码讲解详细,对我们学习C++有一定的帮助,需要的可以参考一下
2023-02-07

Objective-C之Category实现分类示例详解

这篇文章主要为大家介绍了Objective-C之Category实现分类示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2022-11-13

实现shallowReadonly和isProxy功能示例详解

这篇文章主要为大家介绍了实现shallowReadonly和isProxy功能示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2022-12-25

基于Pytorch实现分类器的示例详解

这篇文章主要为大家详细介绍了如何基于Pytorch实现两个分类器: softmax分类器和感知机分类器,文中的示例代码讲解详细,需要的可以参考一下
2023-05-16

Android 获取IP和UA实现示例详解

这篇文章主要为大家介绍了Android 获取IP和UA实现示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-03-19

Python实现内存泄露排查的示例详解

一般在python代码块的调试过程中会使用memory-profiler、filprofiler、objgraph等三种方式进行辅助分析,今天这里主要介绍使用objgraph对象提供的函数接口来进行内存泄露的分析,感兴趣的可以了解一下
2023-01-28

SpringBoot整合Redis实现高并发数据缓存的示例讲解

这篇文章主要介绍了SpringBoot整合Redis实现高并发数据缓存,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2023-03-13

C#实现TCP和UDP通信的示例详解

这篇文章主要为大家详细介绍了C#实现TCP和UDP通信的相关知识,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
2023-03-01

VUE使用localstorage和sessionstorage实现登录示例详解

这篇文章主要为大家介绍了VUE使用localstorage和sessionstorage实现登录示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2022-11-13

C#实现文件分割和合并的示例详解

这篇文章主要为大家详细介绍了如何利用C#实现文件分割和合并的功能,文中的示例代码讲解详细,对我们学习C#有一定的帮助,感兴趣的小伙伴可以跟随小编一起了解一下
2022-12-26

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录