什么是监听器?

前言:重新回忆一下,过滤器,拦截器,监听器! 这三者在实际开发中所起到的作用非同小可.实际开发中必须会的知识点.

千里之行,积于跬步

此图是帮助我们更好整理.

WEB监听器

1,什么是监听器?

web监听器是一种Servlet中的特殊的类,它能帮助开发者监听web中特定的对象,例如ServletContext,HttpSession,ServletRequest三者的创建与销毁跟修改.它能再某些动作的前后增加处理,实现监控

2 监听器通常的用途

*监听器利用HttpServletLisener 统计在线人数

*加载初始化信息:利用ServletContextListener

*统计网站的访问量

*实现访问监控

3 监听器的创建

  首先需要创建一个监听器,实现某种接口,例如我想实现一个对在线人数的监控,可以创建如下的监听器:

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中添加:

  <listener>
      <listener-class>com.test.MyListener</listener-class>
  </listener>

在JSP中添加访问人数:

<body>
    在线人数:<%=session.getAttribute("userNumber") %><br/>
</body>

当我使用我的浏览器访问时,执行结果如下:

 

当打开另一个浏览器访问时:

1,监听器的分类

按照监听对象的划分,可以分为三种:

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接口:

public class MyHttpSessionListener implements HttpSessionListener{
    public void sessionCreated(HttpSessionEvent arg0) {

    }
    public void sessionDestroyed(HttpSessionEvent arg0) {

    }
}

ServletRequest监控:对应监控request内置对象的创建和销毁。

  当访问某个页面时,出发一个request请求,执行requestInitialized方法;当页面关闭时,执行requestDestroyed方法。

  实现方式,直接实现ServletRequestListener接口:

public class MyServletRequestListener implements ServletRequestListener{
    public void requestDestroyed(ServletRequestEvent arg0) {

    }
    public void requestInitialized(ServletRequestEvent arg0) {

    }
}

2 按照监听事件划分:

  2.1 监听事件自身的创建和销毁:同上面的按对象划分。

  2.2 监听属性的新增、删除和修改.

  监听属性的新增、删除和修改也是划分成三种,分别针对于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接口:

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接口:

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());
    }

}

2.3 监听对象的状态:

  针对某些POJO类,可以通过实现HttpSessionBindingListener接口,监听POJO类对象的事件。例如:

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接口就是实现钝化与活化事件的监听:

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());
    }
    
}

getAttribute与getParameter的区别

  这部分是对JSP的扩展,经常在JSP或者Servlet中获取数据,那么getAttribute与getParameter有什么区别呢?

  1 从获取到数据的来源来说:

  getAttribtue获取到的是web容器中的值,比如:

  我们在Servlet中通过setAttribute设定某个值,这个值存在于容器中,就可以通过getAttribute方法获取;

 

  getParameter获取到的是通过http传来的值,比如这样一个http请求:

http:localhost:8080/test/test.html?username=xingoo

  还有其他的GET和POST方式,都可以通过getParameter来获取。

  2 从获取到的数据类型来说:

  getAttribute返回的是一个对象,Object。

  getParameter返回的是,前面页面中某个表单或者http后面参数传递的值,是个字符串。

Oracle ASM 的监听器配置是实现数据库与 ASM 实例之间通信的重要环节,尤其在 RAC(Real Application Clusters)环境中更为关键。ASM 实例需要通过监听器来接受来自数据库实例的连接请求,并进行元数据和文件操作的协调。 ### 配置 Oracle ASM 监听器的基本方法 Oracle ASM 使用标准的 Oracle Net Services 来进行网络通信,其监听器配置方式与数据库实例类似,通常使用 `listener.ora` 文件进行定义。该文件一般位于 `$ORACLE_HOME/network/admin` 目录下。 以下是一个典型的 ASM 监听器配置示例: ```plaintext LISTENER_ASM = (DESCRIPTION_LIST = (DESCRIPTION = (ADDRESS = (PROTOCOL = IPC)(KEY = LISTENER_ASM)) (ADDRESS = (PROTOCOL = TCP)(HOST = node1-vip)(PORT = 1521)) ) ) SID_LIST_LISTENER_ASM = (SID_LIST = (SID_DESC = (GLOBAL_DBNAME = +ASM) (ORACLE_HOME = /u01/app/oracle/product/19.0.0/dbhome_1) (SID_NAME = +ASM1) ) ) ``` 此配置中定义了一个名为 `LISTENER_ASM` 的监听器,支持 IPC 和 TCP 协议,并将 ASM 实例注册为 `+ASM1`,其对应的 ORACLE_HOME 指定为安装目录[^5]。 ### 启动与管理监听器 监听器可以通过 `lsnrctl` 命令进行启动、停止和状态查询: ```bash # 启动监听器 lsnrctl start LISTENER_ASM # 查看监听器状态 lsnrctl status LISTENER_ASM # 停止监听器 lsnrctl stop LISTENER_ASM ``` 在 RAC 环境中,每个节点上的 ASM 实例都需要有独立的监听器配置,并且应确保 SCAN(Single Client Access Name)监听器也正确配置以支持客户端连接[^5]。 ### 日志管理与诊断 ASM 实例的监听器日志文件通常保存在 `$ORACLE_BASE/diag/tnslsnr/<hostname>/listener_asm/trace` 目录下,包括文本格式的 trace 文件和 XML 格式的 alert 文件。这些日志信息可用于故障排查和性能监控。可以使用 ADR 工具统一管理这些日志: ```bash adrci show home set home diag/tnslsnr/node1/listener_asm show alert ``` 上述命令可切换到监听器的日志目录并查看告警日志内容。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值