zoukankan      html  css  js  c++  java
  • Listener监听器笔记1

    1.常用的Web事件监听器接口:

        1.ServletContextListener:用于监听Web应用的启动和关闭。

        2.ServletContextAttributeListener:用于监听ServletContext(application)范围内属性的改变。

        3.ServletRequestListener:用于监听用户的请求。 

        4.ServletRequestAttributeListener:用于监听ServletRequest范围(request)内属性的改变。

        5.HttpSessionListener:用于监听用户session的开始和结束。

        6.HttpSessionAttributeListener:用于监听HttpSession范围(session)内属性的改变。


    2.配置Listener
      1.使用@WebListener修饰Listener实现类即可;或者在web.xml中使用<listener.../>元素进行配置。

    3.使用HttpSessionListener示例:
      监听系统的在线用户:实现HttpSessionListener接口的监听器,可以监听每个用户会话的开始和断开,因此可以通过该监听器监听系统的在线用户

      后台监听器代码:

    /**
     * Description:监听在线用户
     * Author: Eleven
     * Date: 2018/1/8 11:17
     */
    @WebListener
    public class OnlineListener implements HttpSessionListener{
    
        //当用户与服务器之间开始session的时候触发该方法
        @Override
        public void sessionCreated(HttpSessionEvent httpSessionEvent) {
    
            HttpSession session = httpSessionEvent.getSession();
            //设置session有效时间为60秒
            session.setMaxInactiveInterval(60);
            System.out.println("创建"+session.getId()+",当前时间:"+System.currentTimeMillis());
            ServletContext application = session.getServletContext();
            //如果是一次新的会话
            if(session.isNew()){
                //获取session中的用户
                String user = (String) session.getAttribute("user");
                user = (user == null) ? "游客":user;
    
                Map<String,String> online = (Map<String, String>) application.getAttribute("online");
                if(online == null){
                    online = new Hashtable<String,String>();
                }
                //将用户在线信息放入map中
                online.put(session.getId(),user);
                application.setAttribute("online",online);
            }
        }
    
        //当用户与服务器之间session断开时触发该方法
        @Override
        public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
    
            System.out.println("用户与服务器之间session结束");
    
            HttpSession session = httpSessionEvent.getSession();
            System.out.println("销毁"+session.getId()+",当前时间:"+System.currentTimeMillis());
    
            ServletContext application = session.getServletContext();
    
            Map<String,String> online = (Map<String, String>) application.getAttribute("online");
            if(online != null){
                //删除该用户在线信息
                online.remove(session.getId());
            }
    
            application.setAttribute("online",online);
    
        }
    }

     页面jsp

    <%@ page import="java.util.Map" %><%--
      Created by IntelliJ IDEA.
      User: Administrator
      Date: 2018/1/4
      Time: 16:46
      To change this template use File | Settings | File Templates.
    --%>
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>用户在线信息</title>
    </head>
    <body>
    
        在线用户:
        <table width="400" border="1">
        <%
            //获取application中的map
            Map<String,String> online = (Map<String, String>) application.getAttribute("online");
            for(String sessionId:online.keySet()){
        %>
            <tr>
                <td><%=sessionId%></td>
                <td><%=online.get(sessionId)%></td>
            </tr>
        <%
            }
        %>
        </table>
    </body>
    </html>

      接下来,在本机启动两个不同的浏览器来模拟三个用户访问该项目,访问online.jsp页面会看到有三个游客在线,如下图:

      至于为什么只用一个浏览器访问该项目的不同资源,却还是只有一个session,而打开不同的浏览器,就会创建session?这个问题,就留着之后在session中去解决哦~~

      虽然通过实现HttpSessionListener接口可以做到监听在线用户信息,但是这样比较粗糙,只能监听到多少人在线,如果要监听每个用户停留在哪个页面,用户访问的ip等信息,则应该使用HttpServletRequest来实现。

    4.使用ServletRequestListener+ServletContextListener示例:

      具体的做法思路:

      写一个类实现ServletRequestListener,这个监听器就负责监听用户的每次请求,当用户请求到达时,将用户请求的sessionId,用户名,用户IP,正在访问的资源,访问时间记录下来。

      写一个类实现ServletContextListener,随web应用的启动而启动,然后在程序里另开一条线程,这个线程每隔一段时间就去检查每条在线的记录,看每条记录的访问时间与当前时间的差是否超过了一个指定值,如果超过了,就将这条在线记录删掉。

      监听用户请求的代码:

    /**
     * Description:监听用户的每次请求
     * Author: Eleven
     * Date: 2018/1/8 15:33
     */
    @WebListener
    public class RequestListener implements ServletRequestListener{
    
        //当用户请求到达,被初始化时触发该方法
        @Override
        public void requestInitialized(ServletRequestEvent servletRequestEvent) {
    
            HttpServletRequest request = (HttpServletRequest) servletRequestEvent.getServletRequest();
            //session会话被创建
            HttpSession session = request.getSession();
            String sessionId = session.getId();
            //获取访问的ip和访问的页面
            String ip = request.getRemoteAddr();
    
            String page = request.getRequestURI();
            System.out.println("当前会话:"+sessionId+",访问ip:"+ip+",访问页面:"+page);
            //获取用户
            String user = (String) session.getAttribute("user");
            //未登录,设为游客
            user = (user == null) ? "游客":user;
    
            //从数据库中查询该sessionId所对应的用户信息
            try {
                DbDao dd = new DbDao("com.mysql.jdbc.Driver","jdbc:mysql://localhost:3306/test","root","123456");
    
                ResultSet rs =  dd.query2("select * from online where sessionId = '"+sessionId+"' ");
                //存在当前sessionId的用户信息,则表明是旧的会话
                if(rs.next()){
                    //修改访问的page以及当前的时间
                    rs.updateString(4,page);
                    rs.updateLong(5,System.currentTimeMillis());
                    rs.updateRow();
                    rs.close();
                }else{
                    //不存在,则存进数据库
                    dd.insert("insert into online values(?,?,?,?,?)",sessionId,user,ip,page,System.currentTimeMillis());
                }
    
                rs.close();
    
    
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
        //当用户请求结束,被销毁时触发该方法
        @Override
        public void requestDestroyed(ServletRequestEvent servletRequestEvent) {
            HttpServletRequest request = (HttpServletRequest) servletRequestEvent.getServletRequest();
            String page = request.getRequestURI();
            System.out.println("访问结束"+page);
        }
    }

      随web应用而启动,用于检测每条在线记录的监听器代码:

    /**
     * Description: 随web项目启动而启动,然后另开一条线程去判断用户是否已经离线
     * Author: Eleven
     * Date: 2018/1/12 16:45
     */
    @WebListener
    public class TimeListener implements ServletContextListener{
    
        //超过该时间没有访问本站,即认为已经离线
        public final int MAX_MILLIS = 1000*10*2;//2分钟
    
        @Override
        public void contextInitialized(ServletContextEvent servletContextEvent) {
            //每1分钟检查一次
            new Timer(1000*60*1,new ActionListener(){
    
                @Override
                public void actionPerformed(ActionEvent e) {
                    try {
                        //从数据库中查询出所有的用户信息
                        DbDao dd = new DbDao("com.mysql.jdbc.Driver","jdbc:mysql://localhost:3306/test","root","123456");
                        ResultSet rs = dd.query("select * from online ");
                        StringBuffer sb = new StringBuffer();
                        sb.append("(");
                        while(rs.next()){
                            if(System.currentTimeMillis()-rs.getLong(5)>MAX_MILLIS){
                                //超过了10分钟
                                sb.append("'");
                                sb.append(rs.getString(1));
                                sb.append("',");
                            }
                        }
                        System.out.println("aa"+sb.toString());
                        //如果有需要删除的记录
                        if(sb.length()>1){
                            sb.setLength(sb.length()-1);
                            sb.append(")");
                            System.out.println(sb.toString());
                            dd.modify("delete from online where sessionId in " +sb.toString());
                            //删除超时的记录
                        }
                        System.out.println("需要移除的"+sb.toString());
                        rs.close();
                        dd.closeConn();
    
                    } catch (Exception e1) {
                        e1.printStackTrace();
                    }
                }
            }).start();
        }
    
        @Override
        public void contextDestroyed(ServletContextEvent servletContextEvent) {
    
        }
    }

      涉及到的数据库的操作的DbDao的代码:

    /**
     * Description:数据库操作
     * Author: Eleven
     * Date: 2018/1/6 9:27
     */
    public class DbDao {
        private Connection conn;
        private String driver;
        private String url;
        private String name;
        private String psw;
    
        public DbDao() {
        }
    
        public DbDao( String driver, String url, String name, String psw) {
            this.driver = driver;
            this.url = url;
            this.name = name;
            this.psw = psw;
        }
    
        //获取数据库连接
        public Connection getConnection() throws Exception{
    
            if(conn == null){
                //注册驱动
                Class.forName(driver);
                //获取连接
                conn = DriverManager.getConnection(url,name,psw);
            }
            return conn;
        }
        
    
        //查询1  动态查询
        public ResultSet query(String sql,Object... args) throws Exception{
    
            //创建Statement
            PreparedStatement pstmt = getConnection().prepareStatement(sql);
            //设置参数
            for(int i=0;i<args.length;i++){
                 pstmt.setObject(i+1,args[i]);
            }
            return pstmt.executeQuery();
        }
    
        //查询2 
        public ResultSet query2(String sql) throws Exception{
            Statement st = getConnection().createStatement(ResultSet.TYPE_SCROLL_SENSITIVE,ResultSet.CONCUR_UPDATABLE);
    
            return st.executeQuery(sql);
        }
    
    
    
        //插入
        public boolean insert(String sql,Object... args) throws Exception{
            PreparedStatement pstmt = getConnection().prepareStatement(sql);
            for(int i=0;i<args.length;i++){
                pstmt.setObject(i+1,args[i]);
            }
    
            if(pstmt.executeUpdate() != 1){
                return false;
            }
            return true;
        }
    
        //修改或删除
        public void modify(String sql,Object... args) throws Exception{
            PreparedStatement pstmt = getConnection().prepareStatement(sql);
            for(int i=0;i<args.length;i++){
                pstmt.setObject(i+1,args[i]);
            }
            pstmt.executeUpdate();
            pstmt.close();
        }
    
        //关闭数据库连接
        public void closeConn() throws Exception{
            if(conn != null && !conn.isClosed()){
                conn.close();
            }
        }
    
        public Connection getConn() {
            return conn;
        }
    
        public void setConn(Connection conn) {
            this.conn = conn;
        }
    
        public String getDriver() {
            return driver;
        }
    
        public void setDriver(String driver) {
            this.driver = driver;
        }
    
        public String getUrl() {
            return url;
        }
    
        public void setUrl(String url) {
            this.url = url;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public String getPsw() {
            return psw;
        }
    
        public void setPsw(String psw) {
            this.psw = psw;
        }
    }

      页面jsp代码:

    <%@ page import="java.sql.ResultSet" %>
    <%@ page import="servlet.DbDao" %><%--
      Created by IntelliJ IDEA.
      User: Administrator
      Date: 2018/1/4
      Time: 16:46
      To change this template use File | Settings | File Templates.
    --%>
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>用户在线信息</title>
    </head>
    <body>
    
        在线用户:
        <table width="400" border="1">
        <%
            //从数据库总查询出所有的记录
            DbDao dd = new DbDao("com.mysql.jdbc.Driver","jdbc:mysql://localhost:3306/test","root","123456");
            ResultSet rs = dd.query("select * from online");
            while(rs.next()){
        %>
            <tr>
                <td><%=rs.getString(1)%></td>
                <td><%=rs.getString(2)%></td>
                <td><%=rs.getString(3)%></td>
                <td><%=rs.getString(4)%></td>
            </tr>
        <%
            }
        %>
        </table>
    </body>
    </html>

      最后打开不同的浏览器去访问该项目的不同页面,然后再去访问online2.jsp,可以看到如下图的效果:

  • 相关阅读:
    257. Binary Tree Paths
    324. Wiggle Sort II
    315. Count of Smaller Numbers After Self
    350. Intersection of Two Arrays II
    295. Find Median from Data Stream
    289. Game of Life
    287. Find the Duplicate Number
    279. Perfect Squares
    384. Shuffle an Array
    E
  • 原文地址:https://www.cnblogs.com/eleven258/p/8277523.html
Copyright © 2011-2022 走看看