day22-web開發會話技術04

来源:https://www.cnblogs.com/liyuelian/archive/2022/11/23/16919600.html
-Advertisement-
Play Games

WEB開發會話技術04 14.Session生命周期 14.1生命周期說明 public void setMaxInactiveInterval(int interval):設置session的超時時間(以秒為單位),超過指定的時長,session就會被銷毀。 值為正數的時候,設置session的超 ...


WEB開發會話技術04

14.Session生命周期

14.1生命周期說明

  1. public void setMaxInactiveInterval(int interval):設置session的超時時間(以秒為單位),超過指定的時長,session就會被銷毀。

  2. 值為正數的時候,設置session的超時時長。

  3. 值為負數時,表示永不超時

  4. public int getMaxInactiveInterval()表示獲取session的超時時間

  5. public void invalidate()表示讓當前的session會話立即無效

  6. 如果沒有調用setMaxInactiveInterval(int interval)來指定session的生命時長,Tomcat會以session的預設時長為準,session的預設時長為30分鐘,可以在tomcat目錄的conf目錄下的web.xml中設置。

    image-20221122222712748
  7. Session的生命周期指的是:客戶端兩次請求的最大間隔時長,而不是累積時長。即當客戶端訪問了自己的session,session的生命周期將將從0開始重新計算。(指的是同一個會話兩次請求之間的間隔時間)

    cookie的生命周期指的是累積時長

  8. Tomcat用一個線程來輪詢會話狀態,如果某個會話的空閑時間超過設定的最大值,則將該會話銷毀。

    說明:在存放session對象的map中,會記錄所有session對象的生命周期和session的上次被訪問時間。Tomcat維護的線程每隔一定時間就會去掃描這個map,如果發現有某個session對象的上次被訪問時間已超過了其生命周期,就會將其刪除。如果瀏覽器在對應session對象沒有過期的情況下去訪問該session,那麼這個session的上次訪問時間就會被更新成最新訪問的時間。

14.2案例演示1

案例演示1:session的生命周期

web.xml:

<!--CreateSession2-->
<servlet>
    <servlet-name>CreateSession2</servlet-name>
    <servlet-class>com.li.session.CreateSession2</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>CreateSession2</servlet-name>
    <url-pattern>/createSession2</url-pattern>
</servlet-mapping>

<!--ReadSession2-->
<servlet>
    <servlet-name>ReadSession2</servlet-name>
    <servlet-class>com.li.session.ReadSession2</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>ReadSession2</servlet-name>
    <url-pattern>/readSession2</url-pattern>
</servlet-mapping>

CreateSession2:

package com.li.session;

import javax.servlet.*;
import javax.servlet.http.*;
import java.io.IOException;
import java.io.PrintWriter;

public class CreateSession2 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //System.out.println("CreateSession2 被調用");
        //1.創建session
        HttpSession session = request.getSession();
        System.out.println("CreateSession2 sid= " + session.getId());
        //2.設置生命周期為60秒
        session.setMaxInactiveInterval(60);
        //3.放屬性
        session.setAttribute("u", "jack");
        //4.給瀏覽器發送一個回覆
        response.setContentType("text/html;charset=utf-8");
        PrintWriter writer = response.getWriter();
        writer.print("<h1>創建session成功,設置生命周期為60s</h1>");
        writer.flush();
        writer.close();
    }
}

ReadSession2:

package com.li.session;

import javax.servlet.*;
import javax.servlet.http.*;
import java.io.IOException;
import java.io.PrintWriter;

public class ReadSession2 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //System.out.println("ReadSession2 被調用");
        //1.獲取到session
        HttpSession session = request.getSession();
        System.out.println("ReadSession2 sid= " + session.getId());
        //2.讀取session的屬性
        Object u = session.getAttribute("u");
        if (u != null) {
            System.out.println("讀取到session屬性 u= " + (String) u);
        } else {
            System.out.println("讀取不到session屬性u,說明原來的session已經被銷毀了");
        }
        //3.給瀏覽器發送一個回覆
        response.setContentType("text/html;charset=utf-8");
        PrintWriter writer = response.getWriter();
        writer.print("<h1>讀取session成功</h1>");
        writer.flush();
        writer.close();
    }
}
  1. redeployTomcat,首先在瀏覽器中訪問http://localhost:8080/cs/createSession2創建session,然後在設置的60s生命周期內訪問http://localhost:8080/cs/readSession2讀取session,後臺輸出如下:

    image-20221123160535756
  2. 等待60s後,再次訪問http://localhost:8080/cs/readSession2,後臺輸出如下:

    image-20221123160627142

    可以看到session的id和之前不一樣了,說明伺服器創建了新的session,原來的session因為超過了生命周期已經被銷毀。

  3. 在瀏覽器抓包,也可以看出伺服器返回了一個新的jsessionid值:

    image-20221123160842762
  4. 重新訪問http://localhost:8080/cs/createSession2創建session,然後分別在其30s,70s後訪問http://localhost:8080/cs/readSession2,後臺輸出的sid是一致的,說明session的生命周期的計算不是累積的,而是客戶端兩次請求的最大間隔時長。

    image-20221123161637722

14.2案例演示2

案例演示2:刪除session

web.xml:

<!--DeleteSession-->
<servlet>
    <servlet-name>DeleteSession</servlet-name>
    <servlet-class>com.li.session.DeleteSession</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>DeleteSession</servlet-name>
    <url-pattern>/deleteSession</url-pattern>
</servlet-mapping>

DeleteSession:

package com.li.session;

import javax.servlet.*;
import javax.servlet.http.*;
import java.io.IOException;
import java.io.PrintWriter;

public class DeleteSession extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //System.out.println("DeleteSession 被調用");

        //演示如何刪除session
        HttpSession session = request.getSession();
        session.invalidate();
        //如果要刪除session的某個方法,使用session.removeAttribute

        //給瀏覽器發送一個回覆
        response.setContentType("text/html;charset=utf-8");
        PrintWriter writer = response.getWriter();
        writer.print("<h1>刪除session成功</h1>");
        writer.flush();
        writer.close();
    }
}

redeployTomcat,首先訪問http://localhost:8080/cs/createSession2,創建session,然後訪問http://localhost:8080/cs/deleteSession,刪除此session。這時我們再訪問http://localhost:8080/cs/readSession2讀取session當前的sid,可以發現session已經不再是之前那個session了,說明之前創建的session已經被刪除。

後臺輸出如下:

image-20221123164650579

15.Session經典案例-防止非法進入管理頁面

需求說明:完成防止用戶登錄管理頁面應用案例

image-20221123165722319

說明:

  1. 只要密碼為666666,就認為是登錄成功,用戶名不限制
  2. 如果驗證成功,則進入管理頁面ManageServlet.java,否則進入error.html
  3. 如果用戶直接訪問ManageServlet.java,直接重定向到login.html。即不允許在未驗證的情況下直接訪問管理頁面。

練習

思路:

  1. 首先在loginCheckServlet判斷用戶數據是否合法。如果合法,創建保存一個session,將用戶數據保存到session中,並請求轉發到ManageServlet。如果非法,則請求轉發到error.html。

  2. 在ManageServlet中,首先獲取session。如果該session中有設置的用戶數據,說明在此次請求之前,創建過session,併在伺服器保存了該session對象,即用戶登錄過,因此可以直接訪問管理頁面。否則,就說明此次請求之前沒有創建過session,該session是新創建的,用戶沒有登錄驗證過,就重定向到login.html。

LoginCheckServlet:

package com.li.session.hw;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;

@WebServlet(name = "LoginCheckServlet", urlPatterns = {"/loginCheckServlet"})
public class LoginCheckServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        /**
         * 首先在loginCheckServlet判斷用戶數據是否合法。
         * 1.如果合法,創建一個session,給session設置用戶數據,並直接請求轉發到ManageServlet
         * 2.如果非法,請求轉發到error.html。
         */
        //獲取表單數據
        String username = request.getParameter("username");
        String pwd = request.getParameter("pwd");
        if ("666666".equals(pwd)) {//如果數據合法
            //請求轉發到ManageServlet
            HttpSession session = request.getSession();
            session.setAttribute("username", username);
            //伺服器來解析 /
            request.getRequestDispatcher("/manageServlet").forward(request, response);
        } else {//數據非法,請求轉發到error.html
            //伺服器來解析 /
            request.getRequestDispatcher("/error.html").forward(request, response);
        }
    }
}

ManageServlet:

package com.li.session.hw;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.IOException;
import java.io.PrintWriter;

@WebServlet(name = "ManageServlet", urlPatterns = {"/manageServlet"})
public class ManageServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        /**
         * 在ManageServlet中,首先獲取session。
         * 1.如果該session中有設置的用戶數據,說明在此次請求之前,創建過session,
         *   併在伺服器保存了該session對象,即用戶登錄過,因此可以直接訪問管理頁面。
         * 2.否則,就說明此次請求之前沒有創建過session,該session是新創建的,
         *   用戶沒有登錄驗證過,就重定向到login.html。
         */
        HttpSession session = request.getSession();
        Object username = session.getAttribute("username");
        // username=null 說明是新創建的session,說明該用戶沒有登錄過
        if (username == null) {
            //瀏覽器解析的 /
            response.sendRedirect("/cs/login.html");
            return;
        } else {
            //否則說明瀏覽器有對應的session(即已經登錄驗證過),可以直接訪問管理頁面
            //顯示頁面
            response.setContentType("text/html;charset=utf-8");
            PrintWriter writer = response.getWriter();
            writer.print("<h1>用戶管理頁面</h1><br/>" + "歡迎你,管理員:"
                    + username.toString());
            writer.flush();
            writer.close();
        }
    }
}

error.html:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>登錄失敗</title>
</head>
<body>
<h1>登錄失敗</h1>
<a href="/cs/login.html">點擊返回重新登錄</a>
</body>
</html>

login.html:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>登錄頁面</title>
</head>
<body>
<form action="/cs/loginCheckServlet" method="post">
    用戶名:<input type="text" name="username"/><br/><br/>
    密碼:<input type="password" name="pwd"/><br/>
    <input type="submit" value="登錄"/>
</form>
</body>
</html>
  1. redeployTomcat,在瀏覽器訪問http://localhost:8080/cs/login.html,輸入正確的密碼,成功登錄並顯示頁面。

    image-20221123192641298 image-20221123192713505
  2. 此時如果在新標簽頁地址欄訪問http://localhost:8080/cs/manageServlet,是可以直接顯示頁面的,因為之前已經登錄過了。

  3. 如果沒有登錄就訪問http://localhost:8080/cs/manageServlet,會重定向到登錄頁面,無法直接訪問管理頁面。


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

-Advertisement-
Play Games
更多相關文章
  • 寫在前面的話 大多數標簽在展示文本內容的時候都會預設把文本中的空白和換行符去掉,這的確大大的使得文本的排版更加美觀了,也怎加了區域的利用率,可是就有一些需求是需要原原本本的展示出原汁原味的文本格式。那該如何展示出文本的內在格式呢?無外乎把不該去掉的空格展示出來,把不該合併的換行讓它換行起來。 那麼 ...
  • 遇到了一個 Spring Boot 3 整合 MyBatis 的問題,然後解決了。當然,這其實不是個大問題,只是自己編碼時遇到了,然後總結總結分享一下。如果有遇到類似問題的,可以參考一下。 交代一下背景 最近在熟悉 Spring Boot 3 版本的代碼,開發過程中遇到了一些小坑,不過很快都解決了。 ...
  • 最近在學習做微服務的項目,在參考他人的微服務項目時,發現資料庫表所對應的實體類都會實現Serializable介面,以往做的項目中並沒有遇到過,也沒有實現過這個介面,所以好奇實體類為什麼需要實現該介面,在查閱相關博客後,進行了總結記錄 原文鏈接:【java】java實體類為什麼要實現Serializ ...
  • # 1.列表的格式 # [數據1,數據2,數據3,···] # 列表 可變數據類型 # 列表可以存儲多個數據,數據之間的逗號以英文逗號分隔 # 列表可以存儲不同類型數據,但一般存儲同一數據類型,這樣便於操作 # list_name = [] # 定義了一個空的列表 # 定義了一個有數據的列表 # 可 ...
  • 日常工作中 Map 絕對是我們 Java 程式員高頻使用的一種數據結構,那 Map 都有哪些遍歷方式呢?這篇文章阿粉就帶大家看一下,看看你經常使用的是哪一種。 通過 entrySet 來遍歷 1、通過 for 和 map.entrySet() 來遍歷 第一種方式是採用 for 和 Map.Entry ...
  • 由於博主有很多個python環境,如msys64的python,anaconda3的python和官網下載的python, 當我在vscode運行python,需要安裝對應的包時,用pip安裝,如下 安裝成功了,但是還是沒有找到 原因非常簡單,就是我vscode使用的python環境不是上面那個py ...
  • new ,delete 運算符 int *p =new int; delete p; 看一下彙編代碼 可以看到new 和delete 運算符其實也是 operator運算符重載函數的調用 malloc和new malloc 按位元組開闢記憶體 new在開闢記憶體的時候需要指定類型 new int[10] ...
  • 一.小結 1.迴圈語句有三類:while迴圈,do-while迴圈和for迴圈 2.迴圈中需要重覆執行的語句所構成的整體稱為迴圈體 3.迴圈體執行一次稱為迴圈的一次迭代 4.無限迴圈是指迴圈語句被無限次執行 5.在設計迴圈時,既需要考慮迴圈控制構體,還需要考慮迴圈體 6.while迴圈首先檢查迴圈繼 ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...