一、監聽器介紹
1.1、監聽器的概念
監聽器是一個專門用于對其他對象身上發生的事件或狀態改變進行監聽和相應處理的對象,當被監視的對象發生情況時,立即采取相應的行動。監聽器其實就是一個實現特定接口的普通java程序,這個程序專門用于監聽另一個java對象的方法調用或屬性改變,當被監聽對象發生上述事件后,監聽器某個方法立即被執行。
1.2、監聽器案例——監聽window窗口的事件監聽器
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
package me.gacl.listener.demo; import java.awt.Frame; import java.awt.event.WindowEvent; import java.awt.event.WindowListener; public class Demo1 { /** *java的事件監聽機制 *1、事件監聽涉及到三個組件:事件源、事件對象、事件監聽器 *2、當事件源上發生某一個動作時,它會調用事件監聽器的一個方法,并在調用該方法時把事件對象傳遞進去, * 開發人員在監聽器中通過事件對象,就可以拿到事件源,從而對事件源進行操作。 */ public static void main(String[] args) { Frame f = new Frame(); f.setSize( 400 , 400 ); f.setVisible( true ); //注冊事件監聽器 f.addWindowListener( new WindowListener(){ public void windowActivated(WindowEvent e) { } public void windowClosed(WindowEvent e) { } /** * 當window窗體關閉時就會WindowListener這個監聽器監聽到, * 監聽器就會調用windowClosing方法處理window窗體關閉時的動作 */ public void windowClosing(WindowEvent e) { //通過事件對象e來獲取事件源對象 Frame f = (Frame) e.getSource(); System.out.println(f+ "窗體正在關閉" ); f.dispose(); } public void windowDeactivated(WindowEvent e) { } public void windowDeiconified(WindowEvent e) { } public void windowIconified(WindowEvent e) { } public void windowOpened(WindowEvent e) { } }); } } |
1.3、設計一個可以被別的對象監聽的對象
我們平時做開發的時候,我們是寫監聽器去監聽其他對象,那么我們如果想設計一個對象,讓這個對象可以被別的對象監聽又該怎么做呢,可以按照嚴格的事件處理模型來設計一個對象,這個對象就可以被別的對象監聽,事件處理模型涉及到三個組件:事件源、事件對象、事件監聽器。
下面我們來按照事件處理模型來設計一個Person對象,具體代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
package me.gacl.observer; /** * @ClassName: Person(事件源) * @Description: 設計一個Person類作為事件源,這個類的對象的行為(比如吃飯、跑步)可以被其他的對象監聽 * @author: 孤傲蒼狼 * @date: 2014-9-9 下午9:26:06 * */ public class Person { /** * @Field: listener * 在Person類中定義一個PersonListener變量來記住傳遞進來的監聽器 */ private PersonListener listener; /** * @Method: eat * @Description: 設計Person的行為:吃 * @Anthor:孤傲蒼狼 * */ public void eat() { if (listener != null ) { /** * 調用監聽器的doeat方法監聽Person類對象eat(吃)這個動作,將事件對象Event傳遞給doeat方法, * 事件對象封裝了事件源,new Event(this)中的this代表的就是事件源 */ listener.doeat( new Event( this )); } } /** * @Method: run * @Description: 設計Person的行為:跑 * @Anthor:孤傲蒼狼 * */ public void run() { if (listener != null ) { /** * 調用監聽器的dorun方法監聽Person類對象run(跑)這個動作,將事件對象Event傳遞給doeat方法, * 事件對象封裝了事件源,new Event(this)中的this代表的就是事件源 */ listener.dorun( new Event( this )); } } /** * @Method: registerListener * @Description: 這個方法是用來注冊對Person類對象的行為進行監聽的監聽器 * @Anthor:孤傲蒼狼 * * @param listener */ public void registerListener(PersonListener listener) { this .listener = listener; } } /** * @ClassName: PersonListener(事件監聽器) * @Description: 設計Person類(事件源)的監聽器接口 * @author: 孤傲蒼狼 * @date: 2014-9-9 下午9:28:06 * */ interface PersonListener { /** * @Method: doeat * @Description: 這個方法是用來監聽Person對象eat(吃)這個行為動作, * 當實現類實現doeat方法時就可以監聽到Person類對象eat(吃)這個動作 * @Anthor:孤傲蒼狼 * * @param e */ void doeat(Event e); /** * @Method: dorun * @Description: 這個方法是用來監聽Person對象run(跑)這個行為動作, * 當實現類實現dorun方法時就可以監聽到Person類對象run(跑)這個動作 * @Anthor:孤傲蒼狼 * * @param e */ void dorun(Event e); } /** * @ClassName: Event(事件對象) * @Description:設計事件類,用來封裝事件源 * @author: 孤傲蒼狼 * @date: 2014-9-9 下午9:37:56 * */ class Event { /** * @Field: source * 事件源(Person就是事件源) */ private Person source; public Event() { } public Event(Person source) { this .source = source; } public Person getSource() { return source; } public void setSource(Person source) { this .source = source; } } |
經過這樣的設計之后,Peron類的對象就是可以被其他對象監聽了。測試代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
package me.gacl.observer; public class PersonTest { /** * @Method: main * @Description: 測試Person類 * @Anthor:孤傲蒼狼 * * @param args */ public static void main(String[] args) { // Person p = new Person(); //注冊監聽p對象行為的監聽器 p.registerListener( new PersonListener() { //監聽p吃東西這個行為 public void doeat(Event e) { Person p = e.getSource(); System.out.println(p + "在吃東西" ); } //監聽p跑步這個行為 public void dorun(Event e) { Person p = e.getSource(); System.out.println(p + "在跑步" ); } }); //p在吃東西 p.eat(); //p在跑步 p.run(); } } |
運行結果:
me.gacl.observer.Person@4a5ab2在吃東西
me.gacl.observer.Person@4a5ab2在跑步
二、JavaWeb中的監聽器
2.1、基本概念
JavaWeb中的監聽器是Servlet規范中定義的一種特殊類,它用于監聽web應用程序中的ServletContext, HttpSession和 ServletRequest等域對象的創建與銷毀事件,以及監聽這些域對象中的屬性發生修改的事件。
2.2、Servlet監聽器的分類
在Servlet規范中定義了多種類型的監聽器,它們用于監聽的事件源分別為ServletContext,HttpSession和ServletRequest這三個域對象
Servlet規范針對這三個對象上的操作,又把多種類型的監聽器劃分為三種類型:
•監聽域對象自身的創建和銷毀的事件監聽器。
•監聽域對象中的屬性的增加和刪除的事件監聽器。
•監聽綁定到HttpSession域中的某個對象的狀態的事件監聽器。
2.3、監聽ServletContext域對象的創建和銷毀
ServletContextListener接口用于監聽ServletContext對象的創建和銷毀事件。實現了ServletContextListener接口的類都可以對ServletContext對象的創建和銷毀進行監聽。
•當ServletContext對象被創建時,激發contextInitialized (ServletContextEvent sce)方法。
•當ServletContext對象被銷毀時,激發contextDestroyed(ServletContextEvent sce)方法。
•ServletContext域對象創建和銷毀時機:
創建:服務器啟動針對每一個Web應用創建ServletContext
銷毀:服務器關閉前先關閉代表每一個web應用的ServletContext
范例:編寫一個MyServletContextListener類,實現ServletContextListener接口,監聽ServletContext對象的創建和銷毀
1、編寫監聽器,代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
package me.gacl.web.listener; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; /** * @ClassName: MyServletContextListener * @Description: MyServletContextListener類實現了ServletContextListener接口, * 因此可以對ServletContext對象的創建和銷毀這兩個動作進行監聽。 * @author: 孤傲蒼狼 * @date: 2014-9-9 下午10:26:16 * */ public class MyServletContextListener implements ServletContextListener { @Override public void contextInitialized(ServletContextEvent sce) { System.out.println( "ServletContext對象創建" ); } @Override public void contextDestroyed(ServletContextEvent sce) { System.out.println( "ServletContext對象銷毀" ); } } |
2、在web.xml文件中注冊監聽器
我們在上面的中講到,要想監聽事件源,那么必須將監聽器注冊到事件源上才能夠實現對事件源的行為動作進行監聽,在JavaWeb中,監聽的注冊是在web.xml文件中進行配置的,如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
<? xml version = "1.0" encoding = "UTF-8" ?> < web-app version = "3.0" xmlns = "http://java.sun.com/xml/ns/javaee" xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"> < display-name ></ display-name > < welcome-file-list > < welcome-file >index.jsp</ welcome-file > </ welcome-file-list > <!-- 注冊針對ServletContext對象進行監聽的監聽器 --> < listener > < description >ServletContextListener監聽器</ description > <!--實現了ServletContextListener接口的監聽器類 --> < listener-class >me.gacl.web.listener.MyServletContextListener</ listener-class > </ listener > </ web-app > |
經過這兩個步驟,我們就完成了監聽器的編寫和注冊,Web服務器在啟動時,就會自動把在web.xml中配置的監聽器注冊到ServletContext對象上,這樣開發好的MyServletContextListener監聽器就可以對ServletContext對象進行監聽了。
2.4、監聽HttpSession域對象的創建和銷毀
HttpSessionListener 接口用于監聽HttpSession對象的創建和銷毀
創建一個Session時,激發sessionCreated (HttpSessionEvent se) 方法
銷毀一個Session時,激發sessionDestroyed (HttpSessionEvent se) 方法。
范例:編寫一個MyHttpSessionListener類,實現HttpSessionListener接口,監聽HttpSession對象的創建和銷毀
1、編寫監聽器,代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
package me.gacl.web.listener; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; /** * @ClassName: MyHttpSessionListener * @Description: MyHttpSessionListener類實現了HttpSessionListener接口, * 因此可以對HttpSession對象的創建和銷毀這兩個動作進行監聽。 * @author: 孤傲蒼狼 * @date: 2014-9-9 下午11:04:33 * */ public class MyHttpSessionListener implements HttpSessionListener { @Override public void sessionCreated(HttpSessionEvent se) { System.out.println( se.getSession() + "創建了??!" ); } /* HttpSession的銷毀時機需要在web.xml中進行配置,如下: * <session-config> <session-timeout>1</session-timeout> </session-config> 這樣配置就表示session在1分鐘之后就被銷毀 */ @Override public void sessionDestroyed(HttpSessionEvent se) { System.out.println( "session銷毀了??!" ); } } |
2、在web.xml文件中注冊監聽器
1
2
3
4
5
6
7
8
9
10
|
<!--注冊針對HttpSession對象進行監聽的監聽器--> < listener > < description >HttpSessionListener監聽器</ description > < listener-class >me.gacl.web.listener.MyHttpSessionListener</ listener-class > </ listener > <!-- 配置HttpSession對象的銷毀時機 --> < session-config > <!--配置HttpSession對象的1分鐘之后銷毀 --> < session-timeout >1</ session-timeout > </ session-config > |
當我們訪問jsp頁面時,HttpSession對象就會創建,此時就可以在HttpSessionListener觀察到HttpSession對象的創建過程了,我們可以寫一個jsp頁面觀察HttpSession對象創建的過程。
如下:index.jsp
1
2
3
4
5
6
7
8
9
10
11
12
|
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" %> <!DOCTYPE HTML> < html > < head > < title >HttpSessionListener監聽器監聽HttpSession對象的創建</ title > </ head > < body > 一訪問JSP頁面,HttpSession就創建了,創建好的Session的Id是:${pageContext.session.id} </ body > </ html > |
運行結果如下:
2.5、監聽ServletRequest域對象的創建和銷毀
ServletRequestListener接口用于監聽ServletRequest 對象的創建和銷毀
Request對象被創建時,監聽器的requestInitialized(ServletRequestEvent sre)方法將會被調用
Request對象被銷毀時,監聽器的requestDestroyed(ServletRequestEvent sre)方法將會被調用
ServletRequest域對象創建和銷毀時機:
創建:用戶每一次訪問都會創建request對象
銷毀:當前訪問結束,request對象就會銷毀
范例:編寫一個MyServletRequestListener類,實現ServletRequestListener接口,監聽ServletRequest對象的創建和銷毀
1、編寫監聽器,代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
package me.gacl.web.listener; import javax.servlet.ServletRequestEvent; import javax.servlet.ServletRequestListener; /** * @ClassName: MyServletRequestListener * @Description: MyServletRequestListener類實現了ServletRequestListener接口, * 因此可以對ServletRequest對象的創建和銷毀這兩個動作進行監聽。 * @author: 孤傲蒼狼 * @date: 2014-9-9 下午11:50:08 * */ public class MyServletRequestListener implements ServletRequestListener { @Override public void requestDestroyed(ServletRequestEvent sre) { System.out.println(sre.getServletRequest() + "銷毀了??!" ); } @Override public void requestInitialized(ServletRequestEvent sre) { System.out.println(sre.getServletRequest() + "創建了??!" ); } } |
2、在web.xml文件中注冊監聽器
1
2
3
4
5
|
<!--注冊針對ServletRequest對象進行監聽的監聽器--> < listener > < description >ServletRequestListener監聽器</ description > < listener-class >me.gacl.web.listener.MyServletRequestListener</ listener-class > </ listener > |
測試結果如下:
從運行結果中可以看到,用戶每一次訪問都會創建request對象,當訪問結束后,request對象就會銷毀。
以上就是對監聽器的一些簡單講解。
原文鏈接:http://blog.csdn.net/i10630226/article/details/50433428