Java三大器之監聽器(Listener)的工作原理和程式碼演示
阿新 • • 發佈:2020-08-13
現在來說說Servlet的監聽器Listener,它是實現了javax.servlet.ServletContextListener 介面的伺服器端程式,它也是隨web應用的啟動
而啟動,只初始化一次,隨web應用的停止而銷燬。主要作用是:做一些初始化的內容新增工作、設定一些基本的內容、比如一些引數或者是一些
固定的物件等等。首先來看一下ServletContextListener介面的原始碼:
public abstract interface ServletContextListener extends EventListener{ public abstract void contextInitialized(ServletContextEvent paramServletContextEvent);public abstract void contextDestroyed(ServletContextEvent paramServletContextEvent); }
下面利用監聽器對資料庫連線池DataSource的初始化演示它的使用:ListenerTest.java
import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import org.apache.commons.dbcp.BasicDataSource;/** * 現在來說說Servlet的監聽器Listener,它是實現了javax.servlet.ServletContextListener 介面的 * 伺服器端程式,它也是隨web應用的啟動而啟動,只初始化一次,隨web應用的停止而銷燬。主要作用是:做一些初始化 * 的內容新增工作、設定一些基本的內容、比如一些引數或者是一些固定的物件等等。 * * 示例程式碼:使用監聽器對資料庫連線池DataSource進行初始化 */ public class ListenerTest implements ServletContextListener{// 應用監聽器的銷燬方法 public void contextDestroyed(ServletContextEvent servletContextEvent) { ServletContext servletContext = servletContextEvent.getServletContext(); // 在整個web應用銷燬之前呼叫,將所有應用空間所設定的內容清空 servletContext.removeAttribute("dataSource"); System.out.println("銷燬工作完成..."); } // 應用監聽器的初始化方法 public void contextInitialized(ServletContextEvent servletContextEvent) { // 通過這個事件可以獲取整個應用的空間 // 在整個web應用下面啟動的時候做一些初始化的內容新增工作 ServletContext servletContext = servletContextEvent.getServletContext(); // 設定一些基本的內容;比如一些引數或者是一些固定的物件 // 建立DataSource物件,連線池技術 dbcp BasicDataSource basicDataSource = new BasicDataSource(); basicDataSource.setDriverClassName("com.jdbc.Driver"); basicDataSource.setUrl("jdbc:mysqlocalhost:3306/"); basicDataSource.setUsername("root"); basicDataSource.setPassword("root"); basicDataSource.setMaxActive(10);//最大連線數 basicDataSource.setMaxIdle(5);//最大管理數 //bds.setMaxWait(maxWait); 最大等待時間 // 把 DataSource 放入ServletContext空間中, // 供整個web應用的使用(獲取資料庫連線) servletContext.setAttribute("dataSource", basicDataSource); System.out.println("應用監聽器初始化工作完成..."); System.out.println("已經建立DataSource..."); } }
web.xml中配置如下,很簡單:
<!-- 配置應用監聽器 --> <listener> <listener-class>com.ycq.ListenerTest</listener-class> </listener>
這樣配置好了之後,以後在web應用中就可以通過ServletContext取得BasicDataSource物件,從而獲取與資料庫的連線,提高效能,方便使用。
示例程式碼二:
import java.io.File; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import com.i2f.fsp.deploy.TransactionDeployer; /** * 監聽器隨著專案的啟動而啟動 * */ public class ListenerTest2 implements ServletContextListener{ // 銷燬監聽器 public void contextDestroyed(ServletContextEvent servletContextEvent) { System.out.println("date20161020095500 :" + servletContextEvent.getServletContext()); } public void contextInitialized(ServletContextEvent servletContextEvent) { try{ // 獲取專案跟路徑 String basePath = servletContextEvent.getServletContext().getRealPath("/"); // D:\apache-tomcat-6.0.41\webapps\i2money\ 絕對路徑 System.out.println("basePath20161020094700 :" + basePath); if (!(basePath.endsWith(File.separator))){ basePath = basePath + File.separator; } basePath = basePath + "WEB-INF" + File.separator + "classes" + File.separator; new TransactionDeployer(basePath).deploy(); // D:\apache-tomcat-6.0.41\webapps\i2money\WEB-INF\classes\ System.out.println("basePath20161020094701 :" + basePath); } catch (Exception e){ e.printStackTrace(); System.exit(-1); } } }
示例程式碼三:
import javax.servlet.http.HttpSession; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.springframework.context.ApplicationContext; import org.springframework.web.context.support.WebApplicationContextUtils; public class UserLogoutListener implements HttpSessionListener{ protected final Log log = LogFactory.getLog(super.getClass()); public void sessionCreated(HttpSessionEvent event){ this.log.error("session created. id = " + event.getSession().getId()); } public void sessionDestroyed(HttpSessionEvent event){ this.log.error("session destroyed.id = " + event.getSession().getId()); HttpSession session = event.getSession(); ApplicationContext context = WebApplicationContextUtils.getWebApplicationContext(session.getServletContext()); OnlineUserMonitorClient client = (OnlineUserMonitorClient)context.getBean("onlineUserMonitorClient"); client.afterSessionDestroyed(session); } }
監聽器在實際專案中的應用,監聽器在java web中應用的較多,比如:統計當前線上人數、自定義session掃描器。
--------------------- 應用一:統計當前線上人數 ---------------------
import javax.servlet.ServletContext; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; /** * @description HttpSessionListener監聽器實現統計網站線上人數的功能 */ public class SessionListener implements HttpSessionListener{ public static int TOTAL_ONLINE_USERS = 0; public void sessionCreated(HttpSessionEvent httpSessionEvent) { ServletContext servletContext = httpSessionEvent.getSession().getServletContext(); TOTAL_ONLINE_USERS = (Integer) servletContext.getAttribute("TOTAL_ONLINE_USERS"); // 如果使用者退出,TOTAL_ONLINE_USERS自減1 if(TOTAL_ONLINE_USERS == 0){ servletContext.setAttribute("TOTAL_ONLINE_USERS", 1); } else{ TOTAL_ONLINE_USERS--; servletContext.setAttribute("TOTAL_ONLINE_USERS", TOTAL_ONLINE_USERS); } } public void sessionDestroyed(HttpSessionEvent httpSessionEvent) { ServletContext servletContext = httpSessionEvent.getSession().getServletContext(); TOTAL_ONLINE_USERS = (Integer) servletContext.getAttribute("TOTAL_ONLINE_USERS"); // 如果使用者登入,TOTAL_ONLINE_USERS自增1 if(TOTAL_ONLINE_USERS == 0){ servletContext.setAttribute("TOTAL_ONLINE_USERS", 1); } else{ TOTAL_ONLINE_USERS++; servletContext.setAttribute("TOTAL_ONLINE_USERS", TOTAL_ONLINE_USERS); } } }
--------------------- 應用二:自定義session掃描器 ---------------------
import java.util.LinkedList; import java.util.List; import java.util.Timer; import javax.servlet.ServletContextEvent; import javax.servlet.ServletContextListener; import javax.servlet.http.HttpSession; import javax.servlet.http.HttpSessionEvent; import javax.servlet.http.HttpSessionListener; import jeus.util.concurrent50.Collections; /** * @description 當網站使用者量增加時,session佔用的記憶體會越來越大,這時session的管理,將會是一項很大的 * 系統開銷,為了高效的管理session,我們可以寫一個監聽器,定期清理掉過期的session */ public class SessionScanerListener implements HttpSessionListener,ServletContextListener{ // 建立一個執行緒安全的集合,用來儲存session @SuppressWarnings("unchecked") List<HttpSession> sessionList = Collections.synchronizedList(new LinkedList<HttpSession>()); private Object lock = new Object(); public void sessionCreated(HttpSessionEvent httpSessionEvent) { System.out.println("session 建立成功..."); HttpSession httpSession = httpSessionEvent.getSession(); synchronized (lock){ sessionList.add(httpSession); } } public void sessionDestroyed(HttpSessionEvent httpSessionEvent) { System.out.println("session 銷燬成功..."); } // web應用關閉時觸發contextDestroyed事件 public void contextDestroyed(ServletContextEvent servletContextEvent) { System.out.println("web應用關閉..."); } // web應用啟動時觸發contextInitialized事件 public void contextInitialized(ServletContextEvent servletContextEvent) { System.out.println("web應用初始化..."); // 建立定時器 Timer timer = new Timer(); // 每隔30秒就定時執行任務 timer.schedule(new MyTask(sessionList,lock), 0, 1000*30); } }
import java.util.List; import java.util.ListIterator; import java.util.TimerTask; import javax.servlet.http.HttpSession; /** * 定時器,定義定時任務的具體內容 */ public class MyTask extends TimerTask{ private List<HttpSession> list; // 儲存傳遞過來的鎖 private Object lock; // 構造方法 MyTask(List<HttpSession> list, Object lock){ this.list = list; this.lock = lock; } @Override public void run() { // 考慮到多執行緒的情況,這裡必須要同步 synchronized (lock){ System.out.println("定時器開始執行..."); ListIterator<HttpSession> listIterator = list.listIterator(); while(listIterator.hasNext()){ HttpSession httpSession = listIterator.next(); // httpSession.getLastAccessedTime() = session的最後訪問時間 if(System.currentTimeMillis() - httpSession.getLastAccessedTime() > 1000*30){ // 手動銷燬session httpSession.invalidate(); // 從集合中移除已經被銷燬的session listIterator.remove(); } } } } }
轉 :https://www.cnblogs.com/austinspark-jessylu/p/7453398.html