关于ios的cookie,可能很多人都并不太了解,因为ios内部已经帮我们封装好了,只要是NSURLRequest的请求,ios内部会默认帮我们保存服务器返回的cookie并保存在sandbox下,但是具体是怎么进行保存的呢,我在今天的项目里发现了一个问题。
今天在做项目的时候,本来的项目需求是在登录之后,在主页面获取用户信息,但是退出app,下次登录的时候,用户就不需要登录,可以直接获取到信息,所以我就在登录之前调用获取信息的接口,第一次登录的时候,因为没有登录,所以调用获取信息的接口显示的是401,没有授权,当你第一次登录成功的时候,ios会帮你保存登录成功后后台给你返回的cookie,所以下次登录之前调用获取信息的接口,默认会帮你带上之前保存的cookie,这样就不需要登录并且获取信息成功,从而判断是现实登录页面还是主页(一般登录的cookie会保存30天,之后就会提示用户需要登录)。
然而在这中间,我也发现了一些问题,在你登录成功到主页面后,迅速的commond+r 重新运行项目,这个时候,你依然是获取不到用户信息的,显示的还是401未授权。这时候我就奇怪了,明明登录成功,内部帮我保存了cookie,并且在sandbox下也有cookie的文件,怎么会获取不了数据呢,然后我就做了以下的实验。
首先在appDelegate中打印沙河路径
然后打印本地的cookie
然后调用获取数据的接口
运行程序后,进入沙河文件夹
然后打印本地cookie为空(因为第一次打开程序,本地并没有缓存cookie)
接着调用获取数据的接口,显示的是401未授权
然后过了大概5,6秒,sandbox就会自动多一个Cookies的文件夹,显示大小为236字节,这应该就是内部自动帮我们进行的cookie保存,并在下一次调用接口的自动带上cookie
然后回到主页调用接口的时候在成功回调里打印后台给我们返回的cookie,这个时候是可以获取到数据的,并且返回了rememberMe和sid两条cookie
在获取到数据的时候我就立刻command+r,重新运行程序,按道理说在appDelegate中调用获取数据接口的时候会自动带上cookie,应该是可以获取到数据的,但是并不是这样,依然显示401没有授权,并且打印本地的cookie只有一条sid,那rememberMe怎么不见了?
然后我就继续试了几次,结果依旧是这样
本地依旧只有sid这个cookie
后来我意识到可能是系统内部存储cookie是需要一定时间的,于是下一次登录成功后我没有立刻重新运行程序,而是在停留了7,8秒后再重新运行,这个时候才发现sandbox里面的Cookies文件夹的大小变大了,之前的236字节变成844字节了!
然后在appDelegate中打印的本地cookie也恢复正常了
并且之后调用接口也能正常获取到数据了!
总结一下,系统会自动帮你获取cookie并本地保存,但是这是需要一定时间的(具体是为什么我也不清楚),但是这种情况会导致你的项目会出现一定的bug,因为当你的用户会出现登录后不在app内停留而立刻退出的情况,这样的话,可能过几个小时后再次打开的话又需要登录(一般的app一次登录后在一个月内是不需要再次登录的),这样就很影响用户体验。
而针对这样的情况,我们可以在后台返回的cookie时自己手动将cookie保存,并在appDelegate上手动取出来,这样就可以解决问题了!
NSData* cookiesData = [NSKeyedArchiverarchivedDataWithRootObject: [[NSHTTPCookieStoragesharedHTTPCookieStorage]cookies]];
NSUserDefaults* defaults = [NSUserDefaultsstandardUserDefaults];
[defaultssetObject: cookiesDataforKey:@"kCookie"];
[defaultssynchronize];