java中Servlet监听器的工作原理及示例详解

网友投稿 239 2023-04-02

java中Servlet监听器的工作原理及示例详解

监听器就是一个实现特定接口的普通java程序,这个程序专门用于监听另一个java对象的方法调用或属性改变,当被监听对象发生上述事件后,监听器某个方法将立即被执行。

监听器原理

监听原理

1、存在事件源

2、提供监听器

3、为事件源注册监听器

4、操作事件源,产生事件对象,将事件对象传递给监听器,并且执行监听器相应监听方法

监听器典型案例:监听window窗口的事件监听器

例如:swing开发首先制造Frame**窗体**,窗体本身也是一个显示空间,对窗体提供监听器,监听窗体方法调用或者属性改变:

* 关闭窗体时,调用windowListener 的windowclosing() , 传递windowEvent参数,表示窗体关闭事件对象

* 事件对象操作事件源,获得事件源状态

自定义监听器

以上内容可以用下图解释:

通过person.addPersonListener(new PersonListener(){})使事件源与监听器间产生联系。

事件源(在事件源方法中创建事件对象):

监听器(参数是事件对象)

事件对象(通过事件对象可以获得事件源)

测试方法

对上的匿名类补充:匿名内部类的作用是创建一个实现接口的匿名类对象,含义是创建一个继承自PersonListener的匿名类的对象),通过new表达式返回的引用被自动向上转型为对PersonListener的引用

Servlet监听器

(不需要配置,但是监听器仍需要进行注册)

在Servlet规范中定义了多种类型的监听器,它们用于监听的事件源分别为 ServletContext, HttpSession 和 ServletRequest 这三个域对象。

Servlet监听器分为三大类

1、数据域对象创建和http://销毁监听器

2、数据域对象和属性变更监听器

3、绑定到 HttpSession 域中的某个对象的状态的事件监听器

(一)数据域对象创建销毁监听器 — 监听三个与对象 (三个监听器)

1、ServletContextListener : 用来监听ServletContext对象的创建和销毁

监听创建

监听销毁

* ServletContext对象代表全局唯一对象,每个web工程会产生一个ServletContext,服务器启动创建,服务器关闭销毁

编写监听器

步骤一:编写类实现特定监听器接口

步骤二:注册监听器,不是通过事件源,而是在web.xml 进行配置

(监听器和Servlet、Filter不同,不需要url配置,监听器执行不是由用户访问的,监听器 是由事件源自动调用的)

servletContext域对象何时创建和销毁:

创建:服务器启动针对每一个web应用创建servletcontext 销毁:服务器关闭前先关闭代表每一个web应用的servletContext

ServletContextListener主流应用:

第一个:在服务器启动时,对一些对象进行初始化,并且将对象保存ServletContext数据范围内(因为在监听器内可以获得事件源对象) — 全局数据

例如:创建数据库连接池

第二个:对框架进行初始化 例如:Spring框架初始化通过ServletContextListener (因为监听器代码在服务器启动时执行)

Spring框架(配置文件随服务器启动加载) org.springframework.web.context.ContextLoaderListener

第三个:实现任务调度,启动定时程序 (Timer、TimerTask) 使一个程序,定时执行

比如说每天晚上十二点给过生日的人进行生日祝福,中国移动对账户进行同步,会在服务器使用较少的时间,例如凌晨之类,启动一段程序,进行同步

java.util.Timer 一种线程设施,用于安排以后在后台线程中执行的任务。可安排任务执行一次,或者定期重复执行。

Timer提供了启动定时任务方法 schedule

* schedule(TimerTask task, Date firstTime, long period) 用来在指定一个时间启动定时器,定期循环执行

* schedule(TimerTask task, long delay, long period) 用来在当前时间delay多少毫秒后启动定时器

停止定时器,timer.cancel取消任务

2、HttpSession 数据对象创建和销毁监听器 —– HttpqpdLVJkSessionListener

监听Session对象创建

监听Session对象销毁

Session何时创建:request.getSession()

Session何时销毁:关闭服务器,Session过期,session.invalidate

*Session过期时间通过web.xml配置(tomcat配置文件中),默认时间30分钟

配置:

HttpSession监听器

现有如下jsP页面:

1.jsp

2.jsp

访问1.jsp时会执行监听器原因:因为如果观察jsp的源码,封闭式英语培训jsp会被预处理成.java代码(在tomcat中work文件夹下,参见http://blog.csdn.net/megustas_jjc/article/details/53462025),我们打开这个.java代码的源码:

其中的getSession的实现实际就是request.getSession()

3、HttpServletRequest对象的创建和销毁监听器 —- ServletRequestListener

—-监听request对象创建

监听request对象销毁

Request何时创建:请求发起时创建

Request何时销毁:响应结束时销毁

例如:每次刷新界面都会创建销毁一次

注意(创建销毁次数由请求次数决定):

使用forward —- request创建销毁几次 —– 一次

使用sendRedirect —- request创建销毁两次 (两次请求)

(二)ServletContext/HttpSession/ServletRequest中保存数据 创建、修改、移除监听器

ServletContextAttributeListener 监听ServletContext中属性变化

HttpSessionAttributeListener 监听HttpSession中属性变化

ServletRequestAttributeListener 监听ServletRequest中属性变化

attributeAdded 监听属性添加 —- 当数据范围对象没有该属性,第一次添加时调用执行

attributeRemoved 监听属性移除 —- 从一个数据范围对象删除一个已经存在属性执行

attributeReplaced 监听属性替换 —– 当一个数据范围已经存在一个属性,向数据范围添加相同名称属性触发替换方法

例如,此处我们用HttpSessionAttributeListener举例(ServletContextListener与ServletRequestListener同理):

JSP页面

监听器

注册

注意:获得返回值通过session.getAttribute(se.getName())

(三)被绑定Session对象,自我状态感知监听器

保存在 Session 域中的对象可以有多种状态:绑定到 Session 中;从 Session 域中解除绑定;随 Session 对象持久化到一个存储设备中(钝化);随 Session 对象从一个存储设备中恢复(活化)

被存放Session的Java对象,感知自我四种状态变化

1、被绑定

2、被解除绑定

3、被钝化 —– 数据从内存序列化硬盘

4、被活化 —- 数据从硬盘重新加载回内存

HttpSessionBindingListener实现接口的java对象,感知自己被绑定到Session或者从Session中解除绑定

HttpSessionActivationListener实现接口的java对象,感知从内存被钝化硬盘上,雅思托福的区别从硬盘活化到内存中

实现这两个接口的类不需要 web.xml 文件中进行注册,都是由Session自主完成的,例如在存储对象的时候会自动调用绑定

HttpSessionBindingListener

* 绑定对象方法 —-

* 解除绑定方法 —–、当Session对象销毁时,当中所有绑定对象解除绑定

JSP页面:

HttpSessionActivationListener

* 感知对象被活化

* 感知对象被钝化

使用场景:Session保存数据,很长一段时间没用,但是不能销毁Session对象,不想占用服务器内存资源 —– 钝化(将服务器内存中数据序列化硬盘上)

JSP界面

读取数据

注意

钝化和活化应该由tomcat服务器 自动进行 —- 配置tomcat

配置context有几个位置?

1、tomcat/conf/context.xml 对所有虚拟主机 所有web工程生效

2、tomcat/conf/Catalina/localhost/context.xml 对当前虚拟主机所有web工程生效

3、当前工程/META-INF/context.xml 对当前工程有效

钝化后 it315目录在哪里?在“tomcat/work/Catalina/localhost/项目名”目录中

java对象如果想实现序列化,需要实现Serializable接口(因此上述Bean2实现Serializable接口,才可以被钝化,并之后进行活化并读取)

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Spring整合mybatis实现过程详解
下一篇:JAVA代码实现MongoDB动态条件之分页查询
相关文章

 发表评论

暂时没有评论,来抢沙发吧~