`
liudeh_009
  • 浏览: 243289 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Tomcat的Session过期处理策略

阅读更多

       tomcat容器实现类都继承了ContainerBase类,容器在启动的时候都会调用ContainerBase类的threadStart()方法,threadStart()方法如下:

 

    protected void threadStart() {

        if (thread != null)
            return;
        if (backgroundProcessorDelay <= 0)
            return;

        threadDone = false;
        String threadName = "ContainerBackgroundProcessor[" + toString() + "]";
        thread = new Thread(new ContainerBackgroundProcessor(), threadName);
        thread.setDaemon(true);
        thread.start();
    }

 StandardHost,StandardContext,StandardWrapper类的backgroundProcessorDelay的值都为-1,所以ContainerBackgroundProcessor线程都没有启动.而StandardEngine类在创建的时候 将backgroundProcessorDelay 的值设置为10.所以ContainerBackgroundProcessor线程在StandardEngine里启动,ContainerBackgroundProcessor线程代码如下:

protected class ContainerBackgroundProcessor implements Runnable {

        public void run() {
            while (!threadDone) {
                try {
                    Thread.sleep(backgroundProcessorDelay * 1000L);
                } catch (InterruptedException e) {
                    ;
                }
                if (!threadDone) {
                    Container parent = (Container) getMappingObject();//得到StandardEngine
                    ClassLoader cl = 
                        Thread.currentThread().getContextClassLoader();
                    if (parent.getLoader() != null) {
                        cl = parent.getLoader().getClassLoader();
                    }
                    processChildren(parent, cl);
                }
            }
        }

        protected void processChildren(Container container, ClassLoader cl) {
            try {
                if (container.getLoader() != null) {
                    Thread.currentThread().setContextClassLoader
                        (container.getLoader().getClassLoader());
                }
                container.backgroundProcess();
            } catch (Throwable t) {
                log.error("Exception invoking periodic operation: ", t);
            } finally {
                Thread.currentThread().setContextClassLoader(cl);
            }
            Container[] children = container.findChildren();
            for (int i = 0; i < children.length; i++) {
                if (children[i].getBackgroundProcessorDelay() <= 0) {
                    processChildren(children[i], cl);
                }
            }
        }

    }

 ContainerBackgroundProcessor每隔10秒执行一下线程,线程递归调用StandardEngine,StandardHost,StandardContext,StandardWrapper类的backgroundProcess()方法(backgroundProcess()方法在ContainerBase类中)

backgroundProcess()方法如下:

    public void backgroundProcess() {
        
        if (!started)
            return;

        if (cluster != null) {
            try {
                cluster.backgroundProcess();
            } catch (Exception e) {
                log.warn(sm.getString("containerBase.backgroundProcess.cluster", cluster), e);                
            }
        }
        if (loader != null) {
            try {
                loader.backgroundProcess();
            } catch (Exception e) {
                log.warn(sm.getString("containerBase.backgroundProcess.loader", loader), e);                
            }
        }
        if (manager != null) {//manager的实现类StandardManager
            try {
                manager.backgroundProcess();
            } catch (Exception e) {
                log.warn(sm.getString("containerBase.backgroundProcess.manager", manager), e);                
            }
        }
        if (realm != null) {
            try {
                realm.backgroundProcess();
            } catch (Exception e) {
                log.warn(sm.getString("containerBase.backgroundProcess.realm", realm), e);                
            }
        }
        Valve current = pipeline.getFirst();
        while (current != null) {
            try {
                current.backgroundProcess();
            } catch (Exception e) {
                log.warn(sm.getString("containerBase.backgroundProcess.valve", current), e);                
            }
            current = current.getNext();
        }
        lifecycle.fireLifecycleEvent(Lifecycle.PERIODIC_EVENT, null);
    }

 

   在调用StandContext类的backgroundProcess()方法时, 会调用StandardManager的backgroundProcess()方法,该方法如下:

       

    public void backgroundProcess() {
        count = (count + 1) % processExpiresFrequency;
        if (count == 0)
            processExpires();
    }

 
 processExpiresFrequency的默认值为6,所以每6*10秒会调一次processExpires()方法,processExpires()方法如下:

  

    public void processExpires() {

        long timeNow = System.currentTimeMillis();
        Session sessions[] = findSessions();
        int expireHere = 0 ;
        
        if(log.isDebugEnabled())
            log.debug("Start expire sessions " + getName() + " at " + timeNow + " sessioncount " + sessions.length);
        for (int i = 0; i < sessions.length; i++) {
            if (sessions[i]!=null && !sessions[i].isValid()) {//判断session是否有效,无效则清除
                expireHere++;
            }
        }
        long timeEnd = System.currentTimeMillis();
        if(log.isDebugEnabled())
             log.debug("End expire sessions " + getName() + " processingTime " + (timeEnd - timeNow) + " expired sessions: " + expireHere);
        processingTime += ( timeEnd - timeNow );

    }

  该方法主要判断session是否过期,过期则清除.

 

  

    整个过程看起来有点复杂,简单地来说,就是tomcat服务器在启动的时候初始化了一个守护线程,定期去检查有没有Session过期.过期则清除.

0
1
分享到:
评论
2 楼 liudeh_009 2013-08-30  
besterzhao 写道
冒昧问一句,processExpires() 从哪里看出来“过期则清除”了?你贴出来的这个方法里面貌似没有看到呢?

isValid()方法里面有做清楚操作的
1 楼 besterzhao 2013-07-29  
冒昧问一句,processExpires() 从哪里看出来“过期则清除”了?你贴出来的这个方法里面貌似没有看到呢?

相关推荐

    设定tomcat中session过期时间.txt

    ### 设定Tomcat中Session过期时间的三种方式 在Web开发中,Session管理是确保用户状态跟踪的重要机制之一。Tomcat作为一款广泛使用的Java应用服务器及Servlet容器,提供了多种方式来设定Session的有效时长。这有助...

    session过期处理

    5. **分布式Session**:在分布式系统中,由于多个服务器之间共享不了本地内存中的Session,需要借助外部存储(如Redis、Memcached)来实现Session共享,这时需要处理Session复制、同步以及过期策略。 6. **Session...

    tomcat session存到mencache

    在处理用户会话时,Tomcat默认使用基于内存的session管理,但这可能导致在高并发场景下session数据丢失或者性能瓶颈。为了解决这些问题,可以将Tomcat的session存储机制改为Memcache,这是一种高效、分布式的内存...

    tomcat redis session.rar

    此外,处理好Session过期策略,防止内存占用过高。 综上所述,"Tomcat Redis Session"是一种在分布式环境中保证用户会话一致性的解决方案,通过集成Redis和Nginx,有效地解决了Web应用集群中的Session管理难题。在...

    tomcat-session共享

    总结来说,实现“tomcat-session共享”涉及到对Nginx负载均衡策略的理解,以及如何利用Redis这样的外部存储来实现跨服务器的Session共享。这个过程涉及多个组件的配置,需要综合考虑性能、安全和可用性等因素。

    nginx+tomcat shiro实现多tomcat下session共享

    可以通过设置合适的session过期时间、使用Redis的持久化机制以及监控Redis的性能,来优化session共享的效率和可靠性。 总结来说,"nginx+tomcat shiro实现多tomcat下session共享"是一种常见的分布式系统架构策略,...

    tomcat-redis-session-manager的jar包-包含Tomcat7和Tomcat8

    此外,为了确保session数据的安全性和一致性,Tomcat-Redis-Session-Manager提供了多种策略,如session过期策略、session复制和故障转移。例如,可以设置session的超时时间,当用户长时间无操作时,session将在Redis...

    tomcat-redis-session-manager包集合下载(tomcat8)

    此外,还需要关注性能指标,如Redis的内存使用、网络延迟等,并根据实际需求进行调整,例如设置session过期策略、优化序列化算法等。 综上所述,"tomcat-redis-session-manager包集合"为使用Tomcat8和JDK1.8的...

    Redis+Tomcat+Nginx集群实现Session共享,Tomcat Session共享

    3. **过期策略**:合理设置Session的过期时间,防止内存占用过高,同时需要处理Session失效后的清理工作。 4. **扩展性**:随着业务的增长,可能需要增加更多的Tomcat和Redis实例,这时要考虑集群的扩展性和容错...

    tomcat-redis-session-manager

    通过设置适当的过期策略,还可以防止session数据过度积累,避免内存压力。 值得注意的是,使用`tomcat-redis-session-manager`可能会带来额外的网络延迟,因为每次session操作都需要与Redis通信。因此,在大型...

    Memcached-Session-Manager多tomcat实现session共享配置

    下面将详细介绍Memcached-Session-Manager的五种配置策略以及如何在Tomcat集群中进行设置。 1. **基本配置** Memcached-Session-Manager的基本配置涉及在每个Tomcat实例的`context.xml`或`server.xml`中添加一个`...

    tomcat集群session共享jar(三个)

    例如,可以通过设置合适的session过期策略减少Redis的存储压力,或者使用Redis的持久化机制保障数据安全。 6. **安全性考虑**:因为session中可能包含敏感信息,所以在使用Redis共享session时,要确保Redis服务器的...

    tomcat-redis-session-manager实现session共享 配置文件

    3. **Session过期**:设置Redis的过期时间来实现Session的自动失效,避免内存泄漏。 4. **负载均衡**:由于所有Session数据都在Redis中,无论用户请求哪台服务器,都能找到对应的Session,实现负载均衡下的Session...

    tomcat-session-manager环境构建资源

    同时,优化Session存储和过期策略,以平衡内存使用和用户体验。 7. **故障转移和监控**:在高可用集群中,要确保当某个服务器出现问题时,能自动将流量切换到其他健康服务器,同时通过监控工具持续检查系统性能和...

    Tomcat实现session共享demo

    4. **配置Session复制策略**:在`web.xml`中,可以通过`&lt;distributable/&gt;`标签声明应用可分布式部署,这样Tomcat就会启动Session复制功能。 5. **测试和优化**:部署应用后,通过并发用户进行测试,确保Session在...

    memcached-session-manager 实现 tomcat session共享

    例如,你可以查看 Session 的序列化和反序列化逻辑,了解如何与 Memcached 通信,以及在 Session 过期或被移除时的处理策略。源码阅读有助于了解其内部工作原理,解决可能出现的问题。 **工具使用**: 作为 "工具...

    基于redis实现tomcat session共享与集群

    除了基本的配置,还可以进一步优化Session共享方案,例如设置Session过期策略、考虑Redis的主从复制和哨兵模式提高可用性、或者使用Redis的Cluster模式来支持大规模的分布式环境。 总之,通过结合Tomcat和Redis,...

    nginx+tomcat+memcached服务架构实现session共享所需jar包

    此外,为了优化性能,可以考虑使用更高级的缓存策略,如预热、过期策略以及session复制备份,以应对服务器故障。 总结,"nginx+tomcat+memcached"架构通过将session数据存储在Memcached中,实现了跨服务器的session...

    redis tomcat7 session共享

    2. **优化**: 根据实际情况,可能需要调整Redis的过期策略、并发性能以及网络延迟等。例如,可以设置Session的过期时间,避免过多无用的Session占用内存。 通过以上步骤,我们可以成功地在`Tomcat7`中利用`Redis`...

    tomcat8集群session共享(redis处理)

    "Tomcat8集群session共享(redis处理)"的主题正是针对这一问题,通过集成Redis作为分布式缓存来解决。下面我们将详细探讨这个过程。 首先,我们需要理解Session的概念。Session是在Web应用中用于跟踪用户状态的一...

Global site tag (gtag.js) - Google Analytics