iOS WKWebView Cookie的处理

1.

最近接到一个模块迁移的功能,就是把H5页面的某个功能模块嵌入的App中...其中涉及到一些原生和JS交互,cookie认证的问题。。。我这里涉及的cookie是token的回传,登录验证是native

2.

OK,js和native交互的这个很好解决,最烦人的是cookie问题,当然【UIWebView】是真的‘香’,Cookie问题人家自行就给解决了,代价就是内存增加了100MB+,呵呵。。。

3.

内存问题,我随即更换了【WKWebView】,加载页面很快,内存很亮眼,退出控制器也释放占用,但是,这个‘但是’,一出现‘但是’就没好事儿。。。对吧,这个但是就是H5页面总提示“账户未登录”,登录返回的“token”设置后,“web”不能获取到。。。呵呵。。。

小贴士:

UIWebView 支持全局的cookie可以共享 NSHTTPCookieStorage单利中的cookie而WKWebView不能共享( WKWebView存储cookie的路径和NSHTTPCookieStorage存储cookie的路径目前是不同的 ?)

4.

遇到的问题
  1. NSURLRequset添加"Cookie"

小贴士:

  • NSURLRequset的子类NSMutableURLRequest可以设置【allHTTPHeaderFields】属性的值,可以将cookie的值放在请求头中,这样我们的请求就带了cookie 了
  • NSURLRequset还有一个属性:HTTPShouldHandleCookies;
    看文档介绍,意思是处理cookie管理器中的cookie放入请求头之中
  • 重点:如果设置了allHTTPHeaderFields属性就会覆盖或者说忽略cookie管理器中的cookie,全部使用【allHTTPHeaderFields】中的cookie(如果存在的话)
  • 如果loadRequest需要cookie,那么你需要调用[setValue:@"some cookies" forHTTPHeaderField:@"Cookie"]设置cookie到请求头中
  1. 302重定向,问题【1】的cookie丢失
  2. 如果H5页面涉及ajax请求,问题【1】的cookie丢失

5.

Cookie知识传送门儿

解决问题

1. 针对302问题,不使用【allHTTPHeaderFields】设置cookie,将【HTTPShouldHandleCookies】设置为'YES'(默认就是'YES')

2. ajax问题,向【WKWebView】中注入cookie即,【document.cookie 】

举个栗子🌰:

NSHTTPResponse *httpResp = (NSHTTPResponse*)response;

/// 这一步的cookie 可以存储到 userdefaults 或者 存储到 NSHTTPCookieStorage
NSDictionary *fields = [httpResp allHeaderFields];
NSArray *cookies = [NSHTTPCookie cookiesWithResponseHeaderFields:fields forURL:httpResp.URL];

/// 栗子1. 存储到 NSHTTPCookieStorage, 这步就直接可以和UIWebView共享了
for(NSHTTPCookie *cookie in cookies){
        
  [[NSHTTPCookieStorage sharedHTTPCookieStorage] setCookie:cookie];
}

/// 栗子2. 存储到 userdefaults 随时读取
NSData *data = [NSKeyedArchiver archivedDataWithRootObject:cookies];
if(0 < [data length]){
    
    [[NSUserDefaults standardUserDefaults] setObject:data forKey:@"some key decided by you..."];
    [[NSUserDefaults standardUserDefaults] synchronize];
}

/// 创建cookie注入命令
NSString *js_cmd = @"";
for(NSHTTPCookie *cookie in cookies){
    /// domain 可写可不写,默认是url的domain,path还是尽量要写的,其余的看自己
    js_cmd = [js_cmd stringByAppendingFormat:@"document.cookie = '%@=%@;path=%@';", cookie.name, cookie.value, cookie.path];
}

WKUserContentController *cc = [[WKUserContentController alloc] init];
WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc] init];
config.userContentController = cc;

/// 创建脚本
WKUserScript *js = [[WKUserScript alloc] initWithSource:js_cmd injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:NO];
/// 将cookie注入到web
[cc addUserScript:js];
......
......
/// 创建web
WKWebView *web = [[WKWebView alloc] initWithFrame:frame configuration:config];
......
......
NSURLRequset / NSMutableURLRequest *request = ....
/// 这个值默认就是'YES'
request. HTTPShouldHandleCookies = YES;
[web loadRequest: request];

重新加载页面数据都出来了,这次是真香,呵呵...

小记

iOS 11.0后增加了【WKHTTPCookieStore】类,专门用来管理与之关联的WKWebViewCookie,引用结构如下:

WKWebView -> WKWebViewConfiguration -> WKWebsiteDataStore -> WKHTTPCookieStore

设置cookie

举个栗子🌰:

......
......
WKWebView *web = xxx;
if(@available(iOS 11.0, *)){

  [web.configuration.websiteDataStore.httpCookieStore setCookie:cookie completionHandler:^{

     /// setting cookie done..
     /// 注意这里并非是同步的
   }];
/// 直接运行到这里,要同步的话别用锁,会死锁的,用RunLoop
}
......
......

RunLoop改写为同步返回样例:

......
......
WKWebView *web = xxx;
if(@available(iOS 11.0, *)){

  [web.configuration.websiteDataStore.httpCookieStore setCookie:cookie completionHandler:^{

     /// setting cookie done..
   }];

    while(‘停止条件’){

       [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:‘日期,可以设置超时或永久等待,建议设置超时’];
    }
}
/// after to-do
......
......
获取cookie

获取Cookie也会有不同步的问题大同小异,不过一般业务里并不会主动去获取Cookie

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

推荐阅读更多精彩内容