web.xml中的ServletContextListener
要想了解ServletContextListener,先看看web.xml中的<listener>配置。
一)web.xml中的內容載入順序:
首先能夠肯定的是,載入順序與它們在 web.xml 文件裏的先後順序無關。
即不會由於 filter 寫在 listener 的前面而會先載入 filter。終於得出的結論是:listener -> filter -> servlet
同一時候還存在著這樣一種配置節:context-param,它用於向 ServletContext 提供鍵值對,即應用程序上下文信息。我們的 listener, filter 等在初始化時會用到這些上下文中的信息,那麽 context-param 配置節是不是應該寫在 listener 配置節前呢?實際上 context-param 配置節可寫在任何位置,因此真正的載入順序為:context-param -> listener -> filter -> servlet
對於某類配置節而言,與它們出現的順序是有關的。
以 filter 為例。web.xml 中當然能夠定義多個 filter,與 filter 相關的一個配置節是 filter-mapping,這裏一定要註意,對於擁有同樣 filter-name 的 filter 和 filter-mapping 配置節而言。filter-mapping 必須出如今 filter 之後。否則當解析到 filter-mapping 時。它所相應的 filter-name 還沒有定義。web 容器啟動時初始化每一個 filter
時,是依照 filter 配置節出現的順序來初始化的,當請求資源匹配多個 filter-mapping 時,filter 攔截資源是依照 filter-mapping 配置節出現的順序來依次調用 doFilter() 方法的。
servlet 同 filter 類似 ,此處不再贅述。
由此,能夠看出,web.xml 的載入順序是:context-param -> listener -> filter -> servlet ,而同個類型之間的實際程序調用的時候的順序是依據相應的 mapping 的順序進行調用的。
二)ServletContextListener 是什麽:從上面載入順序分享一下,ServletContextListener 就是一個監聽器。
ServletContext 被 Servlet 程序用來與 Web 容器通信。
比如寫日誌,轉發請求。每個 Web 應用程序含有一個Context,被Web應用內的各個程序共享。
由於Context能夠用來保存資源而且共享,所以我所知道的 ServletContext 的最大應用是Web緩存----把不常常更改的內容讀入內存。所以server響應請求的時候就不須要進行慢速的磁盤I/O了。
ServletContextListener 是 ServletContext 的監聽者,假設 ServletContext 發生變化,如server啟動時 ServletContext 被創建,server關閉時 ServletContext 將要被銷毀。
在JSP文件裏。application 是 ServletContext 的實例,由JSP容器默認創建。Servlet 中調用 getServletContext()方法得到 ServletContext 的實例。
三)ServletContextListener 可以做些什麽:一切你想做的,就是一個監聽器可以做的事。
1.設計緩存:
思路:
緩存的思路大概是:
1)server啟動時,ServletContextListener 的 contextInitialized()方法被調用,所以在裏面創建好緩存。能夠從文件裏或者從數據庫中讀取取緩存內容生成類,用 ervletContext.setAttribute()方法將緩存類保存在 ServletContext 的實例中。
2)程序使用 ServletContext.getAttribute()讀取緩存。
假設是 JSP。使用a pplication.getAttribute()。假設是 Servlet。使用 getServletContext().getAttribute()。
假設緩存發生變化(如訪問計數),你能夠同一時候更改緩存和文件/數據庫。
或者你等 變化積累到一定程序再保存。也能夠在下一步保存。
3)server將要關閉時,ServletContextListener 的 contextDestroyed()方法被調用,所以在裏面保存緩存的更改。將更改後的緩存保存回文件或者數據庫,更新原來的內容。
import User; //my own class import DatabaseManager; // my own class import javax.servlet.ServletContext; import javax.servlet.ServletContextListener; public class MyContextListener implements ServletContextListener { private ServletContext context = null; public void contextInitialized(ServletContextEvent event) { context = event.getServletContext(); User user = DatabaseManager.getUserById(1); context.setAttribute("user1", user); } public void contextDestroyed(ServletContextEvent event) { User user = (User)context.getAttribute("user1"); DatabaseManager.updateUserData(user); this.context = null; } }
2)推斷一個配置文件的編碼格式:
public class InitProperties implements ServletContextListener { private static final Logger logger = Logger .getLogger(InitProperties.class); @Override public void contextDestroyed(ServletContextEvent arg0) { // TODO Auto-generated method stub } /** * 該監聽器僅僅用於初始化時監聽配置文件是否是utf-8編碼 且僅僅在初始化時運行一次 對興許的各種請求無影響 * **/ @Override public void contextInitialized(ServletContextEvent arg0) { //載入文件的編碼格式 查看是否是utf-8類型的 try { String fileUrl = PropertiesHander.class.getResource("/evoucher.conf").toString(); fileUrl = fileUrl.replace("file:",""); File file = new File(fileUrl); InputStream ios = new java.io.FileInputStream(file); byte[] b = new byte[3]; try { ios.read(b); ios.close(); } catch (IOException e) { logger.error("檢測編碼載入時關閉流出現載入異常"); } //默認編碼是 23 23 23 的 相同能夠載入這樣的默認的編碼 if (b[0] == 35 && b[1] == 35 && b[2] == 35) { logger.info(file.getName() + ":編碼為默認編碼"); } //utf-8編碼的首字母是 ef bb bf else if (b[0] == -17 && b[1] == -69 && b[2] == -65) { logger.info(file.getName() + ":編碼為UTF-8"); } else { try{ // logger.error(file.getName() + "文件編碼格式不是UTF-8編碼。請又一次確定編碼格式!"); throw new EVoucherException(file.getName() + "文件編碼格式不是UTF-8編碼,請又一次確定編碼格式!"); } catch(EVoucherException e){ // logger.error(e); EVoucherException.getEVException(null, e, logger); System.exit(0); } } } catch (FileNotFoundException e) { logger.error("檢測編碼時載入文件出現異常"); } } }
四)布署 ServletContextListener
<listener> <listener-class>MyServletContextListener</listener-class> </listener>
<!-- 自己主動監聽配置文件是否是utf-8編碼 --> <listener> <listener-class>assp.evoucher.common.util.InitProperties</listener-class> </listener>
web.xml中的ServletContextListener