Cookie與Session簡介

来源:https://www.cnblogs.com/websharing/archive/2018/03/02/8487472.html
-Advertisement-
Play Games

Cookie與Session簡介 本章介紹重點: 本章重點為大家介紹在Web開發過程當中經常會使用到的Cookie與Session的概念,它們的使用方法和應用場景,以及它們的優點與局限等等。深刻理解與掌握它們的用法,可以更好的幫助我們開發出正確可用的產品。 我們為什麼需要Cookie與Session ...


Cookie與Session簡介

本章介紹重點:

本章重點為大家介紹在Web開發過程當中經常會使用到的Cookie與Session的概念,它們的使用方法和應用場景,以及它們的優點與局限等等。深刻理解與掌握它們的用法,可以更好的幫助我們開發出正確可用的產品。

 

 

我們為什麼需要Cookie與Session:

在開始本章程的介紹之前,我們有必要要問自己一個問題,我們為什麼需要Cookie與Session?它們可以幫助我們解決什麼問題?

讓我們先來瞭解瞭解我們經常使用的HTTP協議是什麼:HTTP是一個屬於應用層的面相對象的應用協議,它具有以下五個特點:

  • 支持客戶端<->伺服器服務
  • 簡單快速
  • 靈活
  • 無連接
  • 無狀態

如何理解HTTP的無狀態特性?用當下流行的一句話來描述就是“人生若只如初見”。HTTP協議處理事務請求沒有記憶能力,對同一個URL的請求沒有上下文的關係,你所發出的每一次HTTP請求都是獨立的,它的執行情況與響應的結果不會受到之前的HTTP請求的影響,當然它也不會幹擾影響後續的HTTP請求與響應,處理HTTP請求的伺服器沒有保存客戶端的狀態

可以設想一下,假如基於這些特性而不做任何的特殊處理與改進,我們訪問網路站點時會是怎樣的體驗?拿網購來說吧。當我們訪問某電子商城的時候,提示需要輸入用戶名以及密碼,我們按要求填寫所需信息成功登陸,琳琅滿目的商品展示在我們的眼前,我看上了一款最新的電子產品,打算點進去看看詳細介紹,可接下的情況卻讓我們大跌眼鏡,由於伺服器沒有我們的登陸狀態,網站竟要求我重新輸入用戶名跟密碼?這將會是一個多麼糟糕的用戶體驗。

很明顯,HTTP的這種無狀態的特性嚴重的阻礙了我們與伺服器的交互,於是兩種用於保持HTTP連接狀態的技術就應用而生了:Cookie與Session,這也是我們本章的介紹重點。

 

 

Cookie:

 Cookie是什麼?

翻閱維基百科以及百度百科,我們不難發現,Cookie 是在 HTTP 協議下,伺服器或腳本用以維護客戶工作站信息的一種方式。它是由 Web 伺服器保存在用戶瀏覽器(客戶端)上的小文本文件,可以用於記錄用戶的活動或者是與記錄與狀態相關的信息。Cookie是由W3C組織提出,最早由NetScape公司開發實現,現已經成為了一種標準,主流瀏覽器都支持Cookie機制。

Cookie的工作機制可以參見下麵的圖示:

正如前面所說,Cookie保存在客戶端電腦,不同的瀏覽器,存放的路徑可能有所不同,如下列舉幾個常見主流瀏覽器的存放位置:

  • IE存放位置:C:\Users\{$UserName}\AppData\Roaming\Microsoft\Windows\Cookies\xxx.txt
  • FireFox存放位置:C:\Users\{$UserName}\AppData\Roaming\Mozilla\Firefox\Profiles\{$xxx.default}\cookies.sqlite
  • Chrome存放位置:C:\Users\{$UserName}\AppData\Local\Google\Chrome\User Data\Default\Cookies

註意:IE瀏覽器中,IE將各個站點的Cookie分別保存為一個txt純文本文件;而Firefox和Chrome是將所有的Cookie都保存在一個文件中,該文件的格式為SQLite3資料庫格式的文件。

 

 Cookie的屬性:

Cookie的常用屬性可以參閱下列表格:

 

 Cookie的Server Side操作:

創建一個新的Cookie,我們可以通過以下的代碼來實現:

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
	Cookie myCookie = new Cookie("username", "WebOpenShare");
	myCookie.setMaxAge(300);
	myCookie.setDomain("www.webopenshare.com");
	myCookie.setPath("/");
	myCookie.setSecure(false);
	response.addCookie(myCookie);
}

運行上述的代碼,我們便可以得到一個名為"username",其對應的值為"WepOpenShare"。(如果你是本地運行Tomcat伺服器,請記得修改你電腦hosts文件)

 

Cookie對象沒有提供直接修改其屬性的方法,要修改原有Cookie的值,我們只能夠通過創建同名的Cookie並添加到response中覆蓋掉原Cookie的設置,如下所示:

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
	// ...
	Cookie myModifiedCookie = new Cookie("username", "MyWebOpenShare");
	myModifiedCookie.setMaxAge(600);
	myModifiedCookie.setDomain("www.webopenshare.com");
	myModifiedCookie.setPath("/");
	myModifiedCookie.setSecure(false);
	response.addCookie(myModifiedCookie);
}

同樣的,Cookie對象也沒有提供直接刪除某一個Cookie的操作方法,想要刪除某一個Cookie,我們只能夠通過新建一個maxAge為0的同名Cookie,把它添加到response中刪除原先的Cookie,具體如下代碼所示:

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
	// ...
	Cookie deleted = new Cookie("username", "MyWebOpenShare");
	deleted.setMaxAge(0);
	deleted.setDomain("www.webopenshare.com");
	deleted.setPath("/");
	deleted.setSecure(false);
	response.addCookie(deleted);
}

註意:修改跟刪除某一個Cookie時,我們要通過創建一個同名Cookie來進行覆蓋,對於這一個同名Cookie,除了它的value跟maxAge屬性可以不同,其餘的所有屬性必須跟要修改的Cookie保持一致,否則將會操作失敗。

 

 Cookie的Client Side操作:

我們知道Cookie是保存在客戶端的,那麼我們可不可以通過一些腳本語言,例如JavaScript在瀏覽器中操作Cookie呢?如果有,我們可不可以像在Server Side操作Cookie一樣,對Cookie進行讀寫/修改甚至是刪除的操作呢?

可以肯定的是,我們可以通過腳本在瀏覽器中操作Cookie,但是出於安全性的考慮,它所允許的操作非常有限,僅限於讀取當前功能變數名稱下的Cookie,不可以修改或者刪除Cookie。通過下麵的代碼我們就可以讀取出當前Domain下的Cookies.

<script>document.write(document.cookie);</script>

 

 Cookie應該存放什麼值:

前面的介紹當中,我們提及Cookie存放在客戶端的文件上,如果客戶端被入侵,那麼這些Cookie文件存在被盜的潛在危險;另一方面,我們都知道,HTTP協議採用的是明文傳輸的方式,如果有人監聽或者是截取了我們的請求,那麼我們通過Cookie傳送給伺服器的信息就就全部曝露在入侵者面前。因此,存放在Cookie裡面的值應當儘量避免個人敏感信息,如果非要存放敏感關鍵信息的時候,也要先經過加密處理之後再存放。

 

 

Session:

Session是什麼?

Session是另外一種用戶保持HTTP連接狀態的解決方案,與Cookie不同,Session存放於伺服器記憶體當中,伺服器採用了一種類似於散列表(Hashtable)的形式來存放Session對象信息。

Session保存在伺服器上,所以它會消耗伺服器的資源,因此,我們不應該在Session裡面存放太多太過複雜厚重的對象跟信息,在高併發的情況之下,這樣做很容易會使得伺服器記憶體溢出使得程式異常或者最終宕機。

Session的機制:

Session的機制,可以參見下圖:

 

Session何時創建?

Session何時被創建呢?是不是我們訪問伺服器就會創建一個Session呢?答案是否定的,只有我們在程式中調用到Session的時候,伺服器才會幫我們創建Session對象。例如在Servlet當中,當我們調用request.getSession()的方法時,伺服器就會創建一個Session對象。但是細心的朋友可能會發現,當我訪問某一個JSP頁面的時候,也會有Session對象生成,此時我並沒有在Servlet或者是JSP當中顯式的使用任何的request.getSession()方法。這裡我特別強調的是顯式這個詞語,其實當我們訪問JSP的時候,JSP編譯成Servlet的時候預設會幫我們添加這樣的代碼:HttpSession session = request.getSession(),這也是為什麼我們可以什麼都不做,就直接在JSP頁面當中使用session.getAttribute()或者是其它的操作。

 

Session何時銷毀?

前面談了Session的創建,接下來我們來看看Session何時被銷毀?當然,關閉伺服器可以保證所有的Session被銷毀。那關閉瀏覽器呢?很多人有一個誤解,覺得只要我關閉了瀏覽器,Session就銷毀了。其實這種理解是不全面的,我們知道,Session是由伺服器創建並維護的,因此銷毀與否須由伺服器來完成,單純關閉端瀏覽器並不會觸發銷毀記憶體的動作,如果我們重新打開瀏覽器,並向伺服器發送之前的Session ID,是可以找回之前的Session的。只有在關閉瀏覽器的同時觸發一條伺服器請求,告訴伺服器銷毀掉記憶體,那才可以做到這樣的效果。

恰恰是由於關閉瀏覽器並不能銷毀記憶體,出於安全性的考量,我們需要為Session來設置一個有效期,當一次訪問距離上次的請求超過這個有效期的話,那之前的Session就會銷毀失效。

 

Cookie被禁用時?

Session的工作機制,預設情況下是需要瀏覽器的Cookie來配合工作的。當伺服器創建一個全新的Session之後,會向瀏覽器發送一個Session ID的信息,通常是以Cookie的形式返回給客戶端,在接下來的請求中,客戶端便會自動在請求當中攜帶上這個Session ID的參數,伺服器根據傳遞上來的Session ID直接從伺服器中查找返回之前的Session對象。但是如果客戶端禁用Cookie呢?

由於Cookie可以被人為的禁止,必須有其它機制以便在Cookie被禁止時仍然能夠把Session ID傳遞迴伺服器。經常被使用的一種技術叫做URL重寫,就是把Session ID直接附加在URL路徑的後面,附加方式也有兩種,一種是作為URL路徑的附加信息,表現形式為http://${domain}:${port}/${uri};SESSIONID=xxxxxxxxxxx;另一種是作為查詢字元串附加在URL後面,表現形式為http://${domain}:${port}/${uri}?SESSIONID=xxxxxxxxxxx

 

Session常用方法:

Session的常用方法總結如下表所示:

 

拋出引子:

部署在同一伺服器上的所有應用之間是不是共用同一個Session?請關註我們,在後續章節跨應用Session共用中我們會為您揭秘。

 

我要小額贊助,鼓勵作者寫出更好的文章:

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 基於互動式視窗下的文件操作(文件名為user_list.cfg): >>> print (open('user_list.cfg','r').read())user1:password1user2:password2 >>> print (open('user_list.cfg','r').read ...
  • 在一個"tarball"中(經過 tar 和 gzip 處理過的文件)備份最後 24 小時之內當前目錄下所有修改的文件。 程式代碼如下: ~~~~ !/bin/bash BACKUPFILE=backup $(date +%y %m %d) 在備份文件中嵌入時間。 archive=${1: $BAC ...
  • 去除相同值的元素 Properties的流操作,長久保存 兩種用map記錄單詞或字母個數的方法 ...
  • 在http服務里,header參數和表單參數是經常使用到的,本文主要是練習在Go語言里,如何解析Http請求的header里的參數和表單參數,具體代碼如下: 運行後,在chrom瀏覽器里執行請求:http://127.0.0.1:8001/hello?user=admin&pass=888,服務端會 ...
  • 一個web頁面一定少不了輸入框或者按鈕這兩種元素,那麼在Python里如何使用Selenium操作web頁面里的輸入框和按鈕呢?本文帶你簡單入門。 本文采用了一個例子,就是利用Selenium打開百度網頁,然後進行搜索關鍵字“Python”,執行搜索動作。具體代碼如下: Python Python ...
  • 使用Intellij Idea經常遇到的三種亂碼問題: 1、工程代碼亂碼 2、main方法運行,控制台亂碼 3、tomcat運行,控制台亂碼 解決方案: 1.工程代碼亂碼 Settings > Editor > File Encodings > Global Encodings & Project ...
  • 1、關於AOP AOP(Aspect Oriented Programming),即面向切麵編程,可以說是OOP(Object Oriented Programming,面向對象編程)的補充和完善。OOP引入封裝、繼承、多態等概念來建立一種對象層次結構,用於模擬公共行為的一個集合。OOP允許開發者定 ...
  • 有些人提到C++模板就會下意識地覺得可怕、看不懂、避而遠之。其實模板並不複雜,而且熟練後可以用在日常工作中,可以幫助我們重用代碼,讓代碼更簡潔、易讀、可維護。希望這個系列的文章,能夠讓更多人發現模板的魅力,幫助大家寫出更高質量的代碼。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...