史上最全面『java監聽器』解讀,讀完就能用進項目

java斗帝之路 發佈 2021-08-03T07:42:22.313533+00:00

Web監聽器導圖詳解  監聽器是JAVA Web開發中很重要的內容,其中涉及到的知識,可以參考下面導圖:一、Web監聽器1. 什麼是web監聽器?

Web監聽器導圖詳解

  監聽器是JAVA Web開發中很重要的內容,其中涉及到的知識,可以參考下面導圖:

一、Web監聽器

1. 什麼是web監聽器?

  web監聽器是一種Servlet中的特殊的類,它們能幫助開發者監聽web中的特定事件,比如ServletContext,HttpSession,ServletRequest的創建和銷毀;變量的創建、銷毀和修改等。可以在某些動作前後增加處理,實現監控。

2. 監聽器常用的用途

  通常使用Web監聽器做以下的內容:

  統計在線人數,利用HttpSessionLisener

  加載初始化信息:利用ServletContextListener

  統計網站訪問量

  實現訪問監控

3. 接下來看看一個監聽器的創建以及執行過程

  首先需要創建一個監聽器,實現某種接口,例如我想實現一個對在線人數的監控,可以創建如下的監聽器:

public class MyListener implements HttpSessionListener{
    private int userNumber = 0;
    public void sessionCreated(HttpSessionEvent arg0) {
        userNumber++;
        arg0.getSession().setAttribute("userNumber", userNumber);
    }
    public void sessionDestroyed(HttpSessionEvent arg0) {
        userNumber--;
        arg0.getSession().setAttribute("userNumber", userNumber);
    }
}

  然後在web.xml中配置該監聽器,在web-app中添加:

  <listener>
      <listener-class>com.test.MyListener</listener-class>
  </listener>

  在JSP中添加訪問人數:

<body>
    在線人數:<%=session.getAttribute("userNumber") %><br/>
</body>

  當我使用我的瀏覽器訪問時,執行結果如下:

  當打開另一個瀏覽器訪問時:

  由於打開另一個瀏覽器訪問,相當於另一個會話,因此在線人數會增加。

  對於3.0版本的Servlet來說,還支持使用註解的方式進行配置。

  那麼接下來看看都有哪些監聽器以及方法吧!

二、監聽器的分類

1. 按照監聽的對象劃分:

  按照監聽對象的不同可以劃分為三種:

  ServletContext監控:對應監控application內置對象的創建和銷毀。

  當web容器開啟時,執行contextInitialized方法;當容器關閉或重啟時,執行contextDestroyed方法。

  實現方式:直接實現ServletContextListener接口:

public class MyServletContextListener implements ServletContextListener{
    public void contextDestroyed(ServletContextEvent sce) {

    }
    public void contextInitialized(ServletContextEvent sce) {

    }
}

  HttpSession監控:對應監控session內置對象的創建和銷毀。

  當打開一個新的頁面時,開啟一個session會話,執行sessionCreated方法;當頁面關閉session過期時,或者容器關閉銷毀時,執行sessionDestroyed方法。

  實現方式:直接實現HttpSessionListener接口:

public class MyHttpSessionListener implements HttpSessionListener{
    public void sessionCreated(HttpSessionEvent arg0) {

    }
    public void sessionDestroyed(HttpSessionEvent arg0) {

    }
}

  ServletRequest監控:對應監控request內置對象的創建和銷毀。

  當訪問某個頁面時,出發一個request請求,執行requestInitialized方法;當頁面關閉時,執行requestDestroyed方法。

  實現方式,直接實現ServletRequestListener接口:

public class MyServletRequestListener implements ServletRequestListener{
    public void requestDestroyed(ServletRequestEvent arg0) {

    }
    public void requestInitialized(ServletRequestEvent arg0) {

    }
}

2. 按照監聽事件劃分:

  2.1 監聽事件自身的創建和銷毀:同上面的按對象劃分。

  2.2 監聽屬性的新增、刪除和修改:

  監聽屬性的新增、刪除和修改也是劃分成三種,分別針對於ServletContext、HttpSession、ServletRequest對象:

  ServletContext,實現ServletContextAttributeListener接口:

  通過調用ServletContextAttribtueEvent的getName方法可以得到屬性的名稱。

public class MyServletContextAttrListener implements ServletContextAttributeListener {
        public void attributeAdded( ServletContextAttributeEvent hsbe )
        {
                System.out.println( "In servletContext added :name = " + hsbe.getName() );
        }


        public void attributeRemoved( ServletContextAttributeEvent hsbe )
        {
                System.out.println( "In servletContext removed :name = " + hsbe.getName() );
        }


        public void attributeReplaced( ServletContextAttributeEvent hsbe )
        {
                System.out.println( "In servletContext replaced :name = " + hsbe.getName() );
        }
}

  HttpSession,實現HttpSessionAttributeListener接口:

public class MyHttpSessionAttrListener implements HttpSessionAttributeListener {
        public void attributeAdded( HttpSessionBindingEvent hsbe )
        {
                System.out.println( "In httpsession added:name = " + hsbe.getName() );
        }


        public void attributeRemoved( HttpSessionBindingEvent hsbe )
        {
                System.out.println( "In httpsession removed:name = " + hsbe.getName() );
        }


        public void attributeReplaced( HttpSessionBindingEvent hsbe )
        {
                System.out.println( "In httpsession replaced:name = " + hsbe.getName() );
        }
}

  ServletRequest,實現ServletRequestAttributeListener接口:

public class MyServletRequestAttrListener implements ServletRequestAttributeListener {
        public void attributeAdded( ServletRequestAttributeEvent hsbe )
        {
                System.out.println( "In servletrequest added :name = " + hsbe.getName() );
        }


        public void attributeRemoved( ServletRequestAttributeEvent hsbe )
        {
                System.out.println( "In servletrequest removed :name = " + hsbe.getName() );
        }


        public void attributeReplaced( ServletRequestAttributeEvent hsbe )
        {
                System.out.println( "In servletrequest replaced :name = " + hsbe.getName() );
        }
}

  2.3 監聽對象的狀態:

  針對某些POJO類,可以通過實現HttpSessionBindingListener接口,監聽POJO類對象的事件。例如:

public class User implements HttpSessionBindingListener,Serializable{

    private String username;
    private String password;

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public void valueBound(HttpSessionBindingEvent hsbe) {
        System.out.println("valueBound name: "+hsbe.getName());
    }

    public void valueUnbound(HttpSessionBindingEvent hsbe) {
        System.out.println("valueUnbound name: "+hsbe.getName());
    }

}

  Session數據的鈍化與活化:

  由於session中保存大量訪問網站相關的重要信息,因此過多的session數據就會伺服器性能的下降,占用過多的內存。因此類似資料庫對象的持久化,web容器也會把不常使用的session數據持久化到本地文件或者數據中。這些都是有web容器自己完成,不需要用戶設定。

  不用的session數據序列化到本地文件中的過程,就是鈍化;

  當再次訪問需要到該session的內容時,就會讀取本地文件,再次放入內存中,這個過程就是活化。

  類似的,只要實現HttpSeesionActivationListener接口就是實現鈍化與活化事件的監聽:

public class User implements HttpSessionBindingListener,
HttpSessionActivationListener,Serializable{

    private String username;
    private String password;

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public void valueBound(HttpSessionBindingEvent hsbe) {
        System.out.println("valueBound name: "+hsbe.getName());
    }

    public void valueUnbound(HttpSessionBindingEvent hsbe) {
        System.out.println("valueUnbound name: "+hsbe.getName());
    }

    public void sessionDidActivate(HttpSessionEvent hsbe) {
        System.out.println("sessionDidActivate name: "+hsbe.getSource());
    }

    public void sessionWillPassivate(HttpSessionEvent hsbe) {
        System.out.println("sessionWillPassivate name: "+hsbe.getSource());
    }

}

三、Servlet版本與Tomcat版本

  首先看一下Tomcat官網給出的匹配:


  如果版本不匹配,那麼tomcat是不能發布該工程的,首先看一下版本不匹配時,會發生什麼!

  我試圖創建一個web工程,並且選取了Servlet3.0版本:


  然後我想要在tomcat6中發布,可以看到報錯了!

  JDK版本不對....這是在平時開發如果對Servlet不熟悉的web新手,常犯的錯誤。

解決方法:

  1 在創建時,直接發布到Tomcat容器中,此時Servlet僅僅會列出Tomcat支持的版本:

  2 修改工程Servlet版本配置信息,文件為:工作目錄\SessionExample.settings\org.eclipse.wst.common.project.facet.core.xml

<?xml version="1.0" encoding="UTF-8"?>
<faceted-project>
  <runtime name="Apache Tomcat v6.0"/>
  <fixed facet="java"/>
  <fixed facet="wst.jsdt.web"/>
  <fixed facet="jst.web"/>
  <installed facet="java" version="1.7"/>
  <installed facet="jst.web" version="2.5"/>
  <installed facet="wst.jsdt.web" version="1.0"/>
</faceted-project>

四、getAttribute與getParameter的區別

  這部分是對JSP的擴展,經常在JSP或者Servlet中獲取數據,那麼getAttribute與getParameter有什麼區別呢?

1. 從獲取到數據的來源來說:

  getAttribtue獲取到的是web容器中的值,比如:

  我們在Servlet中通過setAttribute設定某個值,這個值存在於容器中,就可以通過getAttribute方法獲取;

  getParameter獲取到的是通過http傳來的值,比如這樣一個http請求:

http:localhost:8080/test/test.html?username=xingoo

  還有其他的GET和POST方式,都可以通過getParameter來獲取。

2. 從獲取到的數據類型來說:

  getAttribute返回的是一個對象,Object。

  getParameter返回的是,前面頁面中某個表單或者http後面參數傳遞的值,是個字符串。

原文:https://juejin.cn/post/6911973816218435597

關鍵字: