大家好,又见面了,我是你们的朋友全栈君。
观察者模式
Listener的原理是基于观察者模式的,所谓观察者模式简单来说,就是当被观察者的特定事件被触发(一般这某些方法被调用)后,会通知观察者(调用观察者的方法),观察者可以在自己的方法中来对事件做一些处理。 在我们的JavaWeb中,观察者就是Listener,而被观察者可能有三个ServletContext、HttpSession、ServletRequest。而事件指的就是这些对象的创建、修改和删除等。
- 什么是web监听器? web监听器是一种Servlet中的特殊的类,它们能帮助开发者监听web中的特定事件,比如ServletContext,HttpSession,ServletRequest的创建和销毁;变量的创建、销毁和修改等。可以在某些动作前后增加处理,实现监控。
- 监听器常用的用途 统计在线人数,利用HttpSessionLisener 加载初始化信息:利用ServletContextListener 统计网站访问量 实现访问监控
- 监听器的创建以及执行过程 个对在线人数的监控,可以创建如下的监听器:
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中添加:
代码语言:javascript复制<listener>
<listener-class>com.test.MyListener</listener-class>
</listener>
在JSP中添加访问人数:
代码语言:javascript复制<body>
在线人数:<%=session.getAttribute("userNumber") %><br/>
</body>
当我使用我的浏览器访问时,执行结果如下:
当打开另一个浏览器访问时:
监听器的分类
- 按照监听的对象划分: 按照监听对象的不同可以划分为三种: 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接口:
代码语言:javascript复制public class MyHttpSessionListener implements HttpSessionListener{
public void sessionCreated(HttpSessionEvent arg0) {
}
public void sessionDestroyed(HttpSessionEvent arg0) {
}
}
ServletRequest监控:对应监控request内置对象的创建和销毁。 当访问某个页面时,出发一个request请求,执行requestInitialized方法;当页面关闭时,执行requestDestroyed方法。 实现方式,直接实现ServletRequestListener接口:
代码语言:javascript复制public class MyServletRequestListener implements ServletRequestListener{
public void requestDestroyed(ServletRequestEvent arg0) {
}
public void requestInitialized(ServletRequestEvent arg0) {
}
}
- 按照监听事件划分: 监听事件自身的创建和销毁:同上面的按对象划分。 监听属性的新增、删除和修改也是划分成三种,分别针对于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接口:
代码语言:javascript复制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接口:
代码语言:javascript复制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());
}
}
- 监听对象的状态:
针对某些POJO类,可以通过实现HttpSessionBindingListener接口,监听POJO类对象的事件。例如:
代码语言:javascript复制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接口就是实现钝化与活化事件的监听:
代码语言:javascript复制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());
}
}
发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/106671.html原文链接:https://javaforall.cn