从春运坐绿皮火车来说会话管理

前言

话说那时候还是学生,有一年春运,冒着风雪在北京北站买回家的票。悲催的是,特快车的票没买着,只买到了绿皮火车,L 打头的临时车。更悲催的是,原本33个小时的车程(特快不到20个小时),结果……走走停停超过了40多个小时,那煎熬,感觉永无尽头!

绿皮火车
临客火车票

换个别的方式吧!于是我提前下车了,下车到了出站口,车站检票人员瞄了一眼我的车牌,眼神有点奇怪,估计是在想这小伙子是不是傻。还好我的票到站点是更远的站,不需要补票。其他没票的,就被嚷嚷着补票了——那个时候为了能回家,多少人是偷偷混上车的啊! 以前是一把心酸一把泪,现在回忆起来那都是年少时难得的经历啊!

挤火车

火车票与会话

火车票就好像我们后台的会话 Session一样,坐火车的这个过程就是我们的会话过程。我们需要先买票才能上车,下车出站票就失效了。本篇我们就来介绍Dio如何使用会话Session保持登录状态。首先请更新后台代码:https://gitee.com/island-coder/express-api,涉及到的接口如下:

先拿 Postman 走一遍流程做一下接口自测(我们的口号是真正的全栈:产品->设计->后端->前端->测试->运维一条龙服务)。过程见下图:

会话调试

可以看到整个会话过程如下:

  • 登录成功后,后端会返回 Cookie,对应的是一个 sessionId 字段(字段名由后端定),这里就是登录人的会话信息。
  • 每次请求相同域名的接口时候,Postman就会携带这个 Cookie 到后端,从而实现了会话过程的免登录验证。
  • 如果退出登录,后端会清除会话信息,此时携带原有的会话信息请求会被后端认为是无效的。

这个过程对于我们使用 Postman 是无感知的,就好比我们用浏览器访问网页一样。但是对于 App 来说就不一样了。App本身是不会自动把登录会话信息携带到其他接口的,也就是说如果我们登录成功后不做会话处理,那么其他涉及到需要会话信息的接口全部会失败!这就好比我们买了张火车票,但是弄丢了,傻乎乎地跑到检票口才发现进不了站!!!

携带会话信息

怎么办?首先是在买票环节我们要把票保存好,在我们的 HttpUtil 中找一个严实的口袋保存车票,这个口袋就是 Dio 的 options.headers。当我们往 options 里存放信息时,每次请求都会携带这些信息到后端(除非请求本身将其参数覆盖)。

static void setCookie(String cookie) {
  _dioInstance.options.headers['Cookie'] = cookie;
}

options 是 Dio 的默认请求配置,是一个BaseOptions对象,包括了很多属性,例如请求方法、响应类型、请求内容类型,连接超时时间、默认查询参数、请求头等等,通过设置 options 我们可以设置很多默认的参数,从而避免到处设置。常见的设置有:

  • 连接超时:connectTimeout,可以根据需要设置超时时长。
  • 请求内容类型:contentType,是一个字符串,默认是:application/json; charset=utf-8。可以在 headers 里设置也可以指定。
  • 响应类型:responseType,是一个枚举,默认是 json
  • 默认查询参数:queryParameters,一个Map 对象,假设接口有默认的请求参数(如终端类型,版本号这类)可以加入到这里。
  • 请求头:headers,也就是我们今天的主角,可以设置请求的 cookie 信息或者其他参数(比如 JWT 的 token,后端要求传递的其他通用参数)。其中 cookie 固定使用 Cookie 字段存储,当然如果后端要自定义别的字段也是可以的。

找着了口袋,我们买完票后就需要把票存放起来,也就是登录成功后要获取到后端设置的 cookie,这个是在响应头里,调试的时候可以打印出来整个 headers 查看,如下所示:

headers 信息

实际 cookie 存在response.headers.map['set-cookie']中,注意这是一个数组,我们这里因为只有一个 cookie,因此取第一个元素再调用 HttpUtilsetCookie即可。登录处理业务逻辑如下:

_handleSubmit(String username, String password) async {
  EasyLoading.showInfo('请稍候...');
  var response = await AuthService.login(username, password);
  if (response != null && response.statusCode == 200) {
    EasyLoading.showSuccess('登录成功');
    if (response.headers.map['set-cookie'] != null) {
      HttpUtil.setCookie(response.headers.map['set-cookie'][0]);
    }
    Navigator.of(context).pop();
  } else {
    EasyLoading.showInfo(response.statusMessage);
  }
  EasyLoading.dismiss();
}

这样,主要我们不退出登录,我们就可以携带会话信息与后端友好地交互了——碰到查票的你也不用心慌了!

验票

清除会话信息

下车了,以前票默认是要回收的,可千万别只是返回到登录页面哦!退出登录时 调用后端的退出登录接口,成功后需要清除掉本地的存储的sessionclearCookie方法很简单,只是将 headersCookie 设置为 null 即可。

void _logout() async {
  var response = await AuthService.logout();
  if (response != null && response.statusCode == 200) {
    HttpUtil.clearCookie();
    EasyLoading.showSuccess('已退出登录');
  } else {
    print('logout Failed');
  }
}
static void clearCookie() {
  _dioInstance.options.headers['Cookie'] = null;
}

验票

有了这个,我们就可以像乘务员那样验票了。验票这里只是调用了后端的一个需要会话信息的接口来验证。

void _checkSession() async {
  var response = await AuthService.checkSession();
  if (response != null && response.statusCode == 200) {
    print(response.data);
    EasyLoading.showSuccess('验票通过,持票人:' + response.data['loginUser']);
  } else {
    print('Request Failed');
  }
}

运行效果如下图:

运行结果

总结

本篇介绍了 Dio如何在登录后携带会话信息,以避免其他需要登录鉴权的接口请求失败。需要注意的是,会话信息在退出登录后需要及时清除,同时,会话信息可能还携带失效时间信息,可以根据失效时间来判断是否需要重新登录。另外,Dio官方推荐的 Cookie 管理插件是dio_cookie_managerdio_cookie_manager使用拦截器来管理 Cookie,支持使用 CookieJar 来在内存存放 Cookie,或使用 PersistCookieJar持久化Cookie。

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

推荐阅读更多精彩内容

  • Session和Cookie 由于HTTP协议是无状态的协议,所以服务端需要记录用户的状态时,就需要用某种机制来识...
    林昀熙阅读 489评论 0 1
  • 1、谈谈对http协议的认识流程:1.域名解析域名解析检查顺序为:浏览器自身DNS缓存---》OS自身的DNS缓存...
    Zzmi阅读 684评论 0 0
  • http是无状态的,一次请求结束,连接断开,下次服务器再收到请求,它就不知道这个请求是哪个用户发过来的。当然它知道...
    Blues2013阅读 277评论 0 0
  • Xss - 跨站脚本攻击 跨站脚本攻击(Cross Site Scripting)是指攻击者利用网站程序对用户输入...
    XenoBladeII阅读 497评论 0 0
  • DATA LDAP spring.data.ldap.repositories.enabled = true #是...
    chanyi阅读 5,390评论 0 0