当前位置:首页 > > 架构师社区
[导读]这篇文章其实源于一次我的面试经历。那次我面对是一位老面试官,真的很有东西。那次面试我和他叨叨了两小时....我滴妈我嘴巴都干了真的。他的提问都很有深度,可以说对我的学习之路有很大的帮助。

这篇文章其实源于一次我的面试经历。

那次我面对是一位老面试官,真的很有东西。

那次面试我和他叨叨了两小时....我滴妈我嘴巴都干了真的。

他的提问都很有深度,可以说对我的学习之路有很大的帮助。

我记得有个问题,差不多是面了一个小时的时候他问我:Cookie、Session、Token知道的吧?

我说:知道。

那你从演进的角度来讲讲 Cookie 、Session、Token?

我当时就懵了,单独的说我都清楚,这演进的角度让我一下不知从何说起。

这个面试官会从各个角度去感受我对一个知识点到底是背的,还是有自己理解的。

他提问的方式真的很有东西,他也给我反馈了很多他的理解,相谈甚欢,真的。

就这个问题他从 HTTP 无状态开始慢慢的引导我.....

他的这波引导其实就串联起了这一系列的知识点,零散的东西就被他整理的明明白白。

所以后来的学习我都喜欢找缘由,也就是为什么。

也是我一直强调的要知道:为什么会有这个东西的存在,这个的东西是为了解决什么痛点。

起初是因为我怕面试官再问我这样的问题。

现在是因为就应该这样学。

今儿就来捋捋之前面试官问我的这个题的。

正文

1990 年。

蒂姆·伯纳斯·李创建了 HTTP 协议。

李老的想法是把文档存储在服务器中,谁需要这个文档直接从服务器获取即可。

按照这个思想,当时的需求只有 GET。

并且按照拿文档的思路:拿完了连接就可以断了,也不需要什么交互。

所以 HTTP 起初的设计就是无状态的。

也就是请求和请求之间是没有关联的。

而随着互联网的发展,交互开始兴起。

人们不再满足简单的静态文件获取,各种购物、社交接踵而至。

这意味着服务器需要判断每个请求的发起者是谁,也就是需要状态。

你聊天总得表明你是谁,并且和谁聊吧?不然服务器可不知这聊天信息得发给谁。

你购物总得让服务器知道是谁买了这玩意吧?

总不能你买完了下线,再上线发现你买的东西没了。

这时候就是需要一种技术让请求与请求之间建立起联系,让请求变得有状态。

这技术叫 Cookie,就是一个以 Key-Value 格式存储的小文件,保存在客户端本地

比如登录之后,服务器就能设置 Cookie 返回给浏览器,然后保存在本地。

随便截了个百度的,列出来的就是 key,下拉箭头打开里面就有 value。

之前遇到一位老面试官,问我的问题真的有点东西

之后对百度的请求就可以带着 Cookie 去访问服务器,这里假设 BAIDUID 是用户 ID。

百度的服务器一看原来是这个 ID 啊,就知道是“我”请求了,这就有状态了。

简单地说 Cookie 就是存储在本地的一份文件,每次请求都会带上 Cookie 去访问服务器。

所以把一些用户信息塞到 Cookie 里,这样服务器就能判别是哪个用户的请求了

注意 Cookie 是有域的划分的,来看下这个图:

之前遇到一位老面试官,问我的问题真的有点东西

也就是每个域下面都有各自的 Cookie ,访问不同的网站带属于这个网站的 Cookie ,不会带别人的 Cookie ,不然就乱套了。

之前遇到一位老面试官,问我的问题真的有点东西

但是 Cookie 是明文存储在用户本地,而且带有大量的用户信息这不太安全。

并且每次请求都需要带这么多 Cookie 对带宽来说也不太划算。

Session 就解决了这个问题,Session 就是会话,它有更加广泛的含义,在和 Cookie 这些一起谈论的场景,我们把它狭义化。

Session 就是把用户的会话信息存储在服务端。

然后颁发给客户端一个 sessionId,让客户端之后带着 sessionId 来请求。

这样服务端就可以通过 sessionId 去找到这个用户的信息,从而识别请求。

那客户端是如何带上 sessionId 的?

这个 sessionId 还是按照 Cookie 的形式存储在用户的本地,发起请求的时候带上即可。

之前遇到一位老面试官,问我的问题真的有点东西

但是把这种状态信息存储到服务器中使得服务器就有状态了

一般我们部署在线上的服务器会有多台来做负载均衡,也互相作为 backup。

所以如果 Session 的信息存在某一台机器上,那么当下一次请求被负载分到另一台机器那就找不到这个 Session 信息了。

也就不认得这个请求了,可能的现象就是告诉用户没登录,那用户不就傻了。

我这刚还登录着呢,这就告诉我没登录了?

所以处理方式有 session 复制,就是服务器之间互相同步 session,这样不论请求到哪个服务器都有用户的信息。

不过这复制就冗余了,有额外的开销。

还有一种就是 session sticky,其实就是把你的请求一直粘在某一个服务器上,如果你请求的一开始被指派的是 A 服务器,那么之后的所有请求都只会被指派到 A 服务器上。

但是如果 A 服务器挂了,你的请求还是会被指派到别的服务器上,这样一来用户登录信息还是会丢了。

可以看到复制和 sticky 都有缺陷,所以可以把 session 放到第三方存储,比如 Redis 里。

这样服务器等于又没状态了。

而服务器的无状态意味着可以随意伸缩,服务集群根据流量加几台减几台,很方便。

但是把 session 放第三方存储上只是把这个维护从服务器转嫁到第三方身上。

第三方得保证它的高可用,不然用户登录信息又会丢了。

不过一般而言我们的系统本来就要维护的第三方存储,所以影响不大。

小结一下:Cookie 明文存储在本地不太安全,所以想着把用户状态存在服务端,而 Session 就是将用户状态信息保存在服务端。

就暴露 sessionId 给客户端,这样相对而言安全些,并且也减少了网络流量。

但这样服务端就有状态了,难以扩展。

因此可以把 Session 放到第三方存储上,但是等于状态还是由服务端维护。

Token

其实仔细想想,是不是不需要在服务端存储用户的信息?

只需要一个能代表身份的凭证即可,一个服务端颁发给用户凭证,之后的请求让用户带着这个凭证就行。

就像我们的身份证,就代表我们。

这个凭证里面就包含了用户的信息,有人可能怕凭证被伪造。

没事,把凭证给签名了,这样我们服务器就能验证凭证的真伪。

和别人做不得假身份证一样。

这种凭证叫 Token。

如果一个用户登录了系统,我就返回一个 Token 给他,之后每次请求他带着这个 Token 来就行。

服务器验证了真伪之后拿到 Token 里面的用户信息就知道这个请求是谁发的了。

之前遇到一位老面试官,问我的问题真的有点东西

这样服务器就无状态了,是真的无状态了,当然客户端有状态了。

由客户端来保存 Token ,这样是最合理的,不需要在服务端冗余数据。

有了 Token 之后服务器因为无状态所以可扩展,并且 Token 还能跨应用使用。

比如同一个公司不同应用之间的调用,所有应用只要能识别这个 Token 就都能登录。

一个 Token 就搞定了,不用每个网站都登录一遍,这就是单点登录。

如果是第三方服务提供方也更容易地提供服务,只需要颁发一个 Token 给调用者即可。

Token  简单的说就是一个含有凭证信息的令牌,只要服务器能识别这个令牌就能根据令牌的身份进行相应的响应。

其实这还蕴含了时间换空间的思想,把存储在服务器的用户信息暴露出去,利用签名来验证 Token 的真伪。

这样每次请求都需要耗费时间去验签,不过好处就是不需要存储信息,也就是时间换空间。

最后

其实像 Cookie + Session 除了可扩展还有跨域啊、跨站伪造请求等问题。

像 Token 更加灵活,在移动端等场景也更加的适用。

有关文章所讲的演进看起来好像就是 Cookie => Session =>Token

不是的,这几个东西都很有用,文章只是单从认证这一方面来看罢了

特别推荐一个分享架构+算法的优质内容,还没关注的小伙伴,可以长按关注一下:

之前遇到一位老面试官,问我的问题真的有点东西

之前遇到一位老面试官,问我的问题真的有点东西

之前遇到一位老面试官,问我的问题真的有点东西

长按订阅更多精彩▼

之前遇到一位老面试官,问我的问题真的有点东西

如有收获,点个在看,诚挚感谢

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭