Session

1.sessionId  <机器的SID,当前时间>生成一个sessionId,这是全局唯一的。

2.TimeOut 会话的超时时间,注意,这个值和客户端ZooKeeper对象指定的超时不一定相同

3.TickTime

4.isClosing 当SessionTracker检测到会话已经失效了,就会将这个会话的isClosing标记为true,之后这个会话将不在处理任何新的请求

SessionTracker

SessionTracker负责管理Session的整个会话生命周期。

SessionTracker的创建

如何管理多个session?

ExpireTime1 [session,session,session]

ExpireTime2 [session,session,session]

新创建的Session,其expireTime=curTime+SessionTimeout

leader服务器每隔expirationInterval时间就进行会话超时检查。

客户端向服务器发送请求,服务器就进行一次会话激活

客户端如果在sessionTimeout/3时间内,没有向服务器发送过任何请求,就主动发送一个PING请求,服务器收到

该请求后激活会话。

创建SessionTracker

在org.apache.zookeeper.server.ZooKeeperServer的startup方法中,会创建SessionTracker,然后启动它。

    public void startup() {
if (sessionTracker == null) {
createSessionTracker();
}
//SessionTrackerImpl继承了Thread,因此实际上他也是个线程,这里就是调用start方法执行线程。
startSessionTracker();
setupRequestProcessors(); registerJMX(); synchronized (this) {
running = true;
notifyAll();
}
}
    protected void createSessionTracker() {
sessionTracker = new SessionTrackerImpl(this, zkDb.getSessionWithTimeOuts(),
tickTime, 1);
}

激活会话

 synchronized public boolean touchSession(long sessionId, int timeout) {
if (LOG.isTraceEnabled()) {
ZooTrace.logTraceMessage(LOG,
ZooTrace.CLIENT_PING_TRACE_MASK,
"SessionTrackerImpl --- Touch session: 0x"
+ Long.toHexString(sessionId) + " with timeout " + timeout);
}
//获取Session
SessionImpl s = sessionsById.get(sessionId);
// Return false, if the session doesn't exists or marked as closing
if (s == null || s.isClosing()) {
return false;
}
//计算超时时间点
long expireTime = roundToInterval(System.currentTimeMillis() + timeout);

//表明没有超时
if (s.tickTime >= expireTime) {
// Nothing needs to be done
return true;
}
//将Session从旧的set移动到新的set中
//首先取出tickTime对应的set,然后从set中移除掉Session
SessionSet set = sessionSets.get(s.tickTime);
if (set != null) {
set.sessions.remove(s);
}
//迁移到新的set中
s.tickTime = expireTime;
set = sessionSets.get(s.tickTime);
if (set == null) {
set = new SessionSet();
sessionSets.put(expireTime, set);
}
set.sessions.add(s);
return true;
}

会话超时检查

   @Override
synchronized public void run() {
try {
while (running) {
currentTime = System.currentTimeMillis();
//如果还未超时
if (nextExpirationTime > currentTime) {
this.wait(nextExpirationTime - currentTime);
continue;
}

SessionSet set;
//已经超时了,很粗暴的直接从sessionSets中移走过期的session,这里做的真是太棒了
set = sessionSets.remove(nextExpirationTime);
if (set != null) {
//将每一个过期的session标记为close。
for (SessionImpl s : set.sessions) {
setSessionClosing(s.sessionId);
//注意这里哦,很重要的。这里就是调用ZooKeeperServer.expire来关闭session
expirer.expire(s);
}
}
//计算新的过期时间
nextExpirationTime += expirationInterval;
}
} catch (InterruptedException e) {
LOG.error("Unexpected interruption", e);
}
LOG.info("SessionTrackerImpl exited loop!");
}
    public void expire(Session session) {
long sessionId = session.getSessionId();
LOG.info("Expiring session 0x" + Long.toHexString(sessionId)
+ ", timeout of " + session.getTimeout() + "ms exceeded");
//因为会话已经超时了,所以关闭它
close(sessionId);
}
  private void close(long sessionId) {
submitRequest(null, sessionId, OpCode.closeSession, 0, null, null);
}

清理会话

找出这个session创建的所有临时节点,就是去ZooKeeper内存数据库中,根据sesionID来获取这个session创建的所有的临时节点信息,对每一个节点创建节点删除请求,从内存数据库中移除该会话的临时节点。

将session从SessionTrackerImpl中移除

关闭ServerCnxn。

KeeperException.ConnectionLoseException

客户端捕获到这种异常,只需要简单的等待org.apache.zookeeper.ZooKeeper自动重新连接上一个ZooKeeper机器即可,当重新连上了之后,客户端会受到Sync_Connected通知。

SESSION_EXPIRED

由于客户端在会话超时时间内没有向服务器发送PING,服务器认为会话已经过期,然后就会将其标记为失效了。如果之后客户端重新连接上了某一个机器,那么就会出现会话过期异常了。在这种情况下,只能创建一个新的ZooKeeper对象,建立一个新的会话了。

SessionMovedException

ZooKeeper个人笔记Session管理的更多相关文章

  1. Session管理解决方案笔记

    大型网站Session管理解决方案: 1. web服务器之间的session复制.    优点:方案成熟    缺点:复制的性能开销大 2. 减少session使用,使用客户端存储cookie     ...

  2. how tomcat works 读书笔记九 Session管理

    在看本文之前,请先查阅相关Session与Cookie的资料. 这篇资料不错 http://blog.csdn.net/fangaoxin/article/details/6952954 Catali ...

  3. hibernate学习笔记第七天:二级缓存和session管理

    二级缓存配置 1.导入ehcache对应的三个jar包 ehcache/*.jar 2.配置hibernate使用二级缓存 2.1设置当前环境开始二级缓存的使用 <property name=& ...

  4. ZooKeeper 学习笔记

    ZooKeeper学习笔记 1.   zookeeper基本概念 zookeeper是一个分布式的,开放源码的分布式应用程序协调服务,是hadoop和Habase的重要组件,是为分布式应用提供一致性服 ...

  5. ZooKeeper读书笔记

    <ZooKeeper读书笔记> 1.Zookeeper是什么?Zookeeper是一个典型的分布式数据一致性的解决方案,分布式应用可以基于它实现诸如数据发布/订阅.负载均衡.命名服务.分布 ...

  6. ZooKeeper学习笔记(一)——概述

    zookeeper学习笔记(一)--概述 1. 概述 Zookeeper是一个开源的分布式的,为分布式应用提供协调服务的Apache项目.zookeeper从设计模式的角度来理解:是一个基于观察者设计 ...

  7. Zookeeper学习笔记(中)

    Zookeeper学习笔记(中) Zookeeper的基本原理和基本实现 深入了解ZK的基本原理 ZK的一致性: ZAB 协议: Zookeeper 原子消息广播协议 ZK通过选举保证 leader ...

  8. Zookeeper学习笔记(上)

    Zookeeper学习笔记 本篇主要是一些基本的介绍和API的使用介绍, 有些只是记录了知识点,而没有完全在笔记中详细解释, 需要自行查找资料补充相关概念 主要参考了课程中的内容: Zookeeper ...

  9. How Tomcat works — 八、tomcat中的session管理

    在使用shiro的session的时候感觉对于tomcat中session的管理还不是特别清楚,而且session管理作为tomcat中比较重要的一部分还是很有必要学习的. 目录 概述 session ...

随机推荐

  1. MathType 6.9 介绍安装

    1.介绍 MathType是强大的数学公式编辑器,与常见的文字处理软件和演示程序配合使用,能够在各种文档中加入复杂的数学公式和符号,可用在编辑数学试卷.书籍.报刊.论文.幻灯演示等方面,是编辑数学资料 ...

  2. php.ini 安全配置

    (1) 打开php的安全模式 php的安全模式是个非常重要的内嵌的安全机制,能够控制一些php中的函数,比如system(),同时把很多文件操作函数进行了权限控制,也不允许对某些关键文件的文件,比如/ ...

  3. ps批量处理图片

    刚刚有朋友问,ps咋做批量动作呢,其实特别简单,基本一劳永逸,用尺寸做个例子,大家看看就知道了.

  4. [BZOJ3173][Tjoi2013]最长上升子序列

    [BZOJ3173][Tjoi2013]最长上升子序列 试题描述 给定一个序列,初始为空.现在我们将1到N的数字插入到序列中,每次将一个数字插入到一个特定的位置.每插入一个数字,我们都想知道此时最长上 ...

  5. Types of CQRS

    Types of CQRS By Vladimir Khorikov CQRS is a pretty defined concept. Often, people say that you eith ...

  6. 【krpano】浏览点赞插件(源码+介绍+预览)

    简介 最近几天研究了如何在krpano全景的基础上实现记录浏览量和点赞次数,写了一个插件,方便大家使用. 效果截图如下: 每当有用户打开该全景页面时,浏览量会自动加1: 用户可以主动点击点赞按钮,点击 ...

  7. Resource Acquisition Is Initialization(RAII Idiom)

    原文链接:http://en.wikibooks.org/wiki/More_C%2B%2B_Idioms/Resource_Acquisition_Is_Initialization Intent ...

  8. 【目录】 hadoop2.6.0

    hadoop2.6.0 安装+例子运行 数据丢失问题解决 通过代码运行程序流程 利用Hadoop的 Java API 利用JAVA API 实现数据上传

  9. 【XLL 框架库函数】 TempBool/TempBool12

    创建一个包含了 Boolean 类型的 TRUE 或 FALSE 的 XLOPER/XLOPER12 LPXLOPER TempBool(int b); LPXLOPER12 TempBool12(i ...

  10. 第一天:安装nodejs

    1.首先,下载NodeJs程序.地址 http://www.nodejs.org/download/,选择Windows Installer 64bit 2.下载下来后,狂点下一步,安装在本地硬盘上. ...