Token ,Cookie和Session的區別--學習筆記

在作接口測試時,常常會碰到請求參數爲token的類型,可是可能大部分測試人員對token,cookie,session的區別仍是隻知其一;不知其二。爲此我查閱大量的資料作了以下總結。
此篇文章也許是最全最通俗的關於Token ,Cookie和Session的區別的文章,好好揣摩文章的每個字,也許你會有更深的理解!mysql

cookie 是一個很是具體的東西,指的就是瀏覽器裏面能永久存儲的一種數據,僅僅是瀏覽器實現的一種數據存儲功能。web

cookie由服務器生成,發送給瀏覽器,瀏覽器把cookie以kv形式保存到某個目錄下的文本文件內,下一次請求同一網站時會把該cookie發送給服務器。因爲cookie是存在客戶端上的,因此瀏覽器加入了一些限制確保cookie不會被惡意使用,同時不會佔據太多磁盤空間,因此每一個域的cookie數量是有限的。redis

Session

session 從字面上講,就是會話。這個就相似於你和一我的交談,你怎麼知道當前和你交談的是張三而不是李四呢?對方確定有某種特徵(長相等)代表他就是張三。算法

session 也是相似的道理,服務器要知道當前發請求給本身的是誰。爲了作這種區分,服務器就要給每一個客戶端分配不一樣的「身份標識」,而後客戶端每次向服務器發請求的時候,都帶上這個「身份標識」,服務器就知道這個請求來自於誰了。至於客戶端怎麼保存這個「身份標識」,能夠有不少種方式,對於瀏覽器客戶端,你們都默認採用 cookie 的方式。sql

服務器使用session把用戶的信息臨時保存在了服務器上,用戶離開網站後session會被銷燬。這種用戶信息存儲方式相對cookie來講更安全,但是session有一個缺陷:若是web服務器作了負載均衡,那麼下一個操做請求到了另外一臺服務器的時候session會丟失。數據庫

Token

token的意思是「令牌」,是用戶身份的驗證方式,最簡單的token組成:uid(用戶惟一的身份標識)、time(當前時間的時間戳)、sign(簽名,由token的前幾位+鹽以哈希算法壓縮成必定長的十六進制字符串,能夠防止惡意第三方拼接token請求服務器)。還能夠把不變的參數也放進token,避免屢次查庫瀏覽器

傳統身份驗證

HTTP 是一種沒有狀態的協議,也就是它並不知道是誰是訪問應用。這裏咱們把用戶當作是客戶端,客戶端使用用戶名還有密碼經過了身份驗證,不過下回這個客戶端再發送請求時候,還得再驗證一下。安全

解決的方法就是,當用戶請求登陸的時候,若是沒有問題,咱們在服務端生成一條記錄,這個記錄裏能夠說明一下登陸的用戶是誰,而後把這條記錄的 ID 號發送給客戶端,客戶端收到之後把這個 ID 號存儲在 Cookie 裏,下次這個用戶再向服務端發送請求的時候,能夠帶着這個 Cookie ,這樣服務端會驗證一個這個 Cookie 裏的信息,看看能不能在服務端這裏找到對應的記錄,若是能夠,說明用戶已經經過了身份驗證,就把用戶請求的數據返回給客戶端。服務器

上面說的就是 Session,咱們須要在服務端存儲爲登陸的用戶生成的 Session ,這些 Session 可能會存儲在內存,磁盤,或者數據庫裏。咱們可能須要在服務端按期的去清理過時的 Session 。cookie

基於 Token 的身份驗證

使用基於 Token 的身份驗證方法,在服務端不須要存儲用戶的登陸記錄。大概的流程是這樣的:

  • 客戶端使用用戶名跟密碼請求登陸
  • 服務端收到請求,去驗證用戶名與密碼
  • 驗證成功後,服務端會簽發一個 Token,再把這個 Token 發送給客戶端
  • 客戶端收到 Token 之後能夠把它存儲起來,好比放在 Cookie 裏或者 Local Storage 裏
  • 客戶端每次向服務端請求資源的時候須要帶着服務端簽發的 Token
  • 服務端收到請求,而後去驗證客戶端請求裏面帶着的 Token,若是驗證成功,就向客戶端返回請求的數據

APP登陸的時候發送加密的用戶名和密碼到服務器,服務器驗證用戶名和密碼,若是成功,以某種方式好比隨機生成32位的字符串做爲token,存儲到服務器中,並返回token到APP,之後APP請求時,凡是須要驗證的地方都要帶上該token,而後服務器端驗證token,成功返回所須要的結果,失敗返回錯誤信息,讓他從新登陸。其中服務器上token設置一個有效期,每次APP請求的時候都驗證token和有效期。

那麼個人問題來了:

  1. 服務器上的token存儲到數據庫中,每次查詢會不會很費時。若是不存儲到數據庫,應該存儲到哪裏呢。
  2. 客戶端獲得的token確定要加密存儲的,發送token的時候再解密。存儲到數據庫仍是配置文件呢?

token是個易失數據,丟了無非讓用戶從新登陸一下,新浪微博動不動就讓我從新登陸,反正這事兒我是無所謂啦。
因此若是你以爲普通的數據庫表撐不住了,能夠放到 MSSQL/MySQL 的內存表裏(不過聽說mysql的內存表性能提高有限),能夠放到 Memcache裏(講真,這個是挺常見的策略),能夠放到redis裏(我作過這樣的實現),甚至能夠放到 OpenResty 的變量字典裏(只要你有信心不爆內存)。

token是個憑條,不過它比門票溫柔多了,門票丟了從新花錢買,token丟了從新操做下認證一個就能夠了,所以token丟失的代價是能夠忍受的——前提是你別丟太頻繁,要是讓用戶隔三差五就認證一次那就損失用戶體驗了。

基於這個出發點,若是你認爲用數據庫來保持token查詢時間太長,會成爲你係統的瓶頸或者隱患,能夠放在內存當中。
好比memcached、redis,KV方式很適合你對token查詢的需求。
這個不會太佔內存,好比你的token是32位字符串,要是你的用戶量在百萬級或者千萬級,那纔多少內存。
要是數據量真的大到單機內存扛不住,或者以爲一宕機全丟風險大,只要這個token生成是足夠均勻的,高低位切一下分到不一樣機器上就行,內存絕對不會是問題。

客戶端方面這個除非你有一個很是安全的辦法,好比操做系統提供的隱私數據存儲,那token確定會存在泄露的問題。好比我拿到你的手機,把你的token拷出來,在過時以前就均可以以你的身份在別的地方登陸。
解決這個問題的一個簡單辦法 

一、在存儲的時候把token進行對稱加密存儲,用時解開。

二、將請求URL、時間戳、token三者進行合併加鹽簽名,服務端校驗有效性。

這兩種辦法的出發點都是:竊取你存儲的數據較爲容易,而反彙編你的程序hack你的加密解密和簽名算法是比較難的。然而其實說難也不難,因此終究是防君子不防小人的作法。話說加密存儲一個你要是被人扒開客戶端看也不會被噴明文存儲……

  • 方法1:它拿到存儲的密文解不開、
  • 方法2:它不知道你的簽名算法和鹽,二者能夠結合食用。

可是若是token被人拷走,他天然也能植入到本身的手機裏面,那到時候他的手機也能夠以你的身份來用着,這你就瞎了。
因而能夠提供一個讓用戶能夠主動expire一個過去的token相似的機制,在被盜的時候能遠程止損。

在網絡層面上token明文傳輸的話會很是的危險,因此建議必定要使用HTTPS,而且把token放在post body裏。

原文連接:https://blog.csdn.net/ToBeTheEnder/article/details/52485948

相關文章
相關標籤/搜索