`
zhoushuping
  • 浏览: 45691 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

关于tomcat集群进行负载均衡的问题

阅读更多
在开发大规模网站程序时,一般会让开发人员尽量采用Cookie而非Session来存 储一些状态相关的数据,例如用户登录信息等等。原因是不推荐做Session复制,而多个应用服务器之间为了共享Session所采取的做法无非就是采用 多播技术的Session复制,另外一种则是将Session持久化到数据库。

第二种做法性能很差,第一种做法在应用服务器节点非常多的时候复制的代价很高,实现tomcat集群中比较关键的问题还是session共享的问题。

一般而言对http session有如下的处理方案:
1、在服务器端不保存Session,完全无状态

     对于不需要保持用户状态的Web应用,采用Stateless是最为恰当的,因此就不存在Session共享的问题。REST (Representational State Transfer) 算是最为典型的例子。
2、基于浏览器Cookie的Session共享

      此种方案把用户相关的Session信息存储到浏览器的Cookie中,也称为客户端Session。

      采用Flash Cookie、URL重写的方式传递Session信息的方案也可以归为此类。

      缺点:只能够存储字符串、数值等基本类型的数据;Cookie大小存在限制;安全性;带宽及数据解压缩、网络传输性能问题。
3、基于数据库的Session共享,实现分布式应用间Session共享

     此种方案把Session信息存储到数据库表,这样实现不同应用服务器间Session信息的共享。诸如Websphere Portal、Weblogic Portal都采用了类似的方案。

     Tomcat Persistent Manager 的JDBC Based Store 提供了类似实现机制,表结构如下:

        create table tomcat_sessions (
            session_id     varchar(100) not null primary key,
            valid_session  char(1) not null,
            max_inactive   int not null,
            last_access    bigint not null,
            app_name       varchar(255),
            session_data   mediumblob,
            KEY kapp_name(app_name)
          );

        优点:实现简单

        缺点:由于数据库服务器相对于应用服务器更难扩展且资源更为宝贵,在高并发的Web应用中,最大的性能瓶颈通常在于数据库服务器。因此如果将Session存储到数据库表,频繁的增加、删除、查询操作很容易造成数据库表争用及加锁,最终影响业务。


4、基于应用服务器/Servlet容器的Clustering机制

        一些常用的应用服务器及Servlet容器的Clustering机制可以实现Session Replication的功能,例如Tomcat Clustering/Session Replication、Jboss buddy replication。

         缺点:基于Clustering的Session复制性能很差,扩展性也很不行。
5、基于NFS的Session共享

         通过NFS方式来实现各台服务器间的Session共享,各台服务器只需要mount共享服务器的存储Session的磁盘即可,实现较为简单。但NFS 对高并发读写的性能并不高,在硬盘I/O性能和网络带宽上存在较大瓶颈,尤其是对于Session这样的小文件的频繁读写操作。

        基于磁盘阵列/SAN/NAS等共享存储的方案道理也类似。
6、基于Terracotta、Ehcache、JBossCache等Java Caching方案实现Session共享

    如果系统架构是Java体系,可以考虑采用Terracotta、Ehcache、JbossCache、Oscache等Java Caching方案来实现Session 共享。

    缺点:架构用于非java体系很不方便;对于是诸如静态页面之类的缓存,采用Memcached的方案比Java更为高效
7、基于Memcached/Tokyo Tyrant 等Key-Value DB的Session共享

    整体说来此种方案扩展性最好,推荐使用。

    原理:Tomcat 服务器提供了org.apache.catalina.session.StandardManager 和org.apache.catalina.session.PersistentManager用于Session对象的管理,可以自定义 PersistentManager的

Store 类来实现自己Memcached、Tokyo Tyrant、Redis等Key-Value DB的客户端。

    以Memcached的客户端为例(摘自Use MemCacheStore in Tomcat):

package com.yeeach;

import com.danga.MemCached.MemCachedClient;

import com.danga.MemCached.SockIOPool;

public class MemCacheStore extends StoreBase implements Store {

/**
* The descriptive information about this implementation.
*/
protected static String info = "MemCacheStore/1.0";

/**
* The thread safe and thread local memcacheclient instance.
*/
private static final ThreadLocal<MemCachedClient> memclient = new ThreadLocal<MemCachedClient>();

/**
* The server list for memcache connections.
*/
private List<String> servers = new ArrayList<String>();

/**
* all keys for current request session.
*/
private List<String> keys = Collections
.synchronizedList(new ArrayList<String>());

/**
* Return the info for this Store.
*/
public String getInfo() {
return (info);
}

/**
* Clear all sessions from the cache.
*/
public void clear() throws IOException {
getMemcacheClient().flushAll();
keys.clear();
}

/**
* Return local keyList size.
*/
public int getSize() throws IOException {
return getKeyList().size();
}

/**
* Return all keys
*/
public String[] keys() throws IOException {
return getKeyList().toArray(new String[] {});
}

/**
* Load the Session from the cache with given sessionId.
*
*/
public Session load(String sessionId) throws ClassNotFoundException,
IOException {

try {

byte[] bytes = (byte[]) getMemcacheClient().get(sessionId);
if (bytes == null)
return null;
ObjectInputStream ois = bytesToObjectStream(bytes);

StandardSession session = (StandardSession) manager
.createEmptySession();
session.setManager(manager);
session.readObjectData(ois);
if (session.isValid() && !keys.contains(sessionId)) {
keys.add(sessionId);
}
return session;

} catch (Exception e) {
return (null);
}
}

/**
* transform a vaild Session from objectinputstream.
* Check which classLoader is responsible for the current instance.
*
* @param bytes
* @return ObjectInputStream with the Session object.
* @throws IOException
*/
private ObjectInputStream bytesToObjectStream(byte[] bytes)
throws IOException {
ByteArrayInputStream bais = new ByteArrayInputStream(bytes);
ObjectInputStream ois = null;
Loader loader = null;
ClassLoader classLoader = null;
Container container = manager.getContainer();
if (container != null)
loader = container.getLoader();
if (loader != null)
classLoader = loader.getClassLoader();
if (classLoader != null)
ois = new CustomObjectInputStream(bais, classLoader);
else
ois = new ObjectInputStream(bais);
return ois;
}

/**
* remove the session with given sessionId
*/
public void remove(String sessionId) throws IOException {
getMemcacheClient().delete(sessionId);
List<String> keyList = getKeyList();
keyList.remove(sessionId);
}

/**
* Store a objectstream from the session into the cache.
*/
public void save(Session session) throws IOException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos);
StandardSession standard = (StandardSession) session;
standard.writeObjectData(oos);
getMemcacheClient().add(session.getId(), baos.toByteArray());
Object ob = getMemcacheClient().get(session.getId());
List<String> keyList = getKeyList();
keyList.add(session.getId());
}

/**
*
* @return
*/
private List<String> getKeyList() {
return keys;
}

/**
* Simple instanc of the Memcache client and SockIOPool.
* @return memchacheclient
*/
private MemCachedClient getMemcacheClient() {
if (memclient == null) {

Integer[] weights = { 1 };
// grab an instance of our connection pool
SockIOPool pool = SockIOPool.getInstance();
if (!pool.isInitialized()) {
String[] serverlist = servers.toArray(new String[] {});
// set the servers and the weights
pool.setServers(serverlist);
pool.setWeights(weights);

// set some basic pool settings
// 5 initial, 5 min, and 250 max conns
// and set the max idle time for a conn
// to 6 hours
pool.setInitConn(5);
pool.setMinConn(5);
pool.setMaxConn(250);
pool.setMaxIdle(1000 * 60 * 60 * 6);

// set the sleep for the maint thread
// it will wake up every x seconds and
// maintain the pool size
pool.setMaintSleep(30);

// set some TCP settings
// disable nagle
// set the read timeout to 3 secs
// and don't set a connect timeout
pool.setNagle(false);
pool.setSocketTO(3000);
pool.setSocketConnectTO(0);

// initialize the connection pool
pool.initialize();
}

// lets set some compression on for the client
// compress anything larger than 64k

memclient.get().setCompressEnable(true);
memclient.get().setCompressThreshold(64 * 1024);
}
return memclient.get();
}

public List<String> getServers() {
return servers;
}

public void setServers(String serverList) {
StringTokenizer st = new StringTokenizer(serverList, ", ");
servers.clear();
while (st.hasMoreTokens()) {
servers.add(st.nextToken());
}
}

}

Tomcat 的配置文件:

<Context path="/test" docBase="test.war">
<Manager className="org.apache.catalina.session.PersistentManager"
distributable="true">
<Store className="com.yeeach.MemcachedStore"
servers="192.168.0.111:11211,192.168.0.112:11211" />
</Manager>

</Context>



    这里只是作为测试演示了在Tomcat中集成Memcached的实现方案。并没有考虑性能、高可用、Memcached 存储Session的持久化(可以使用Memcachedb实现)、Session管理等问题。

    针对Tomcat与Memcached集成有一个开源项目memcached-session-manager  功能实现相对完善,尤其是其设计思想值得借鉴。

    The memcached session manager installed in a tomcat holds all sessions locally in the own jvm, just like the StandardManager does it as well.

    Additionally, after a request was finished, the session (only if existing) is additionally sent to a memcached node for backup.

    When the next request for this session has to be served, the session is locally available and can be used, after this second request is finished the session is updated in the memcached node.

目前正在测试Nginx+tomcat来做集群的问题。
分享到:
评论

相关推荐

    Tomcat集群与负载均衡

    Apache Tomcat 集群负载均衡 ##### 1.13 Tomcat 端口配置 为了实现集群中的负载均衡,需要对 Tomcat 实例上的端口进行配置。这通常涉及到以下端口: - **HTTP 端口**:用于接收客户端的 HTTP 请求。 - **AJP 端口*...

    Apache,Tomcat集群和负载均衡

    Apache,Tomcat集群和负载均衡包括了apache-tomcat-5.5.29.zip,apache_2.2.4-win32-x86-no_ssl.msi,mod_jk-1.2.28-httpd-2.2.3.so,ApacheTomcat整合文档.doc,Apache,Tomcat集群和负载均衡教程.doc,Tomcat负载...

    Tomcat集群和负载均衡配置

    Tomcat集群和负载均衡配置.docx server.xml httpd.conf context.xml

    Tomcat集群和负载均衡.txt

    ### Tomcat集群与负载均衡详解 #### 一、前言 在现代Web应用开发中,随着用户数量的增长和业务需求的变化,单台服务器往往难以满足高性能、高可用性的要求。因此,采用多台服务器组成的集群架构成为了一种常见的...

    tomcat集群和负载均衡

    当我们谈论“Tomcat集群和负载均衡”时,我们指的是通过多台Tomcat服务器来分发和处理来自用户的网络请求,以提高系统可用性和性能。 首先,让我们了解什么是Tomcat集群。在集群环境中,多台Tomcat服务器共享相同的...

    Tomcat服务器集群和负载均衡

    本文将深入探讨“Tomcat服务器集群”和“负载均衡”的概念,以及如何进行相关的配置。 首先,我们要理解什么是Tomcat服务器集群。集群是一种通过多台服务器共享工作负载、提供冗余和提高可用性的方式。在Tomcat环境...

    Nginx+tomcat配置集群负载均衡实例

    配置Nginx+Tomcat集群负载均衡的第一步是安装Nginx和多个Tomcat实例。在多台服务器上部署Tomcat,形成一个集群,确保服务的高可用性。每台服务器上的Tomcat实例都需要配置相同的应用,以处理相同类型的请求。 接...

    Apache,Tomcat集群和负载均衡所需软件下载

    在配置Apache-Tomcat集群时,有以下步骤: 1. **安装和配置Apache**:安装Apache服务器并加载mod_jk模块。 2. **配置mod_jk**:编辑`workers.properties`文件,定义每个Tomcat实例的工作节点,包括IP地址、端口和...

    tomcat集群与负载均衡(windows)

    【Tomcat集群与负载均衡(windows)】 在企业级应用中,Tomcat因其轻量级、易用性而被广泛采用。然而,随着系统规模的扩大,单个Tomcat服务器的负载能力可能无法满足需求,这时就需要考虑部署Tomcat集群以实现负载...

    Nginx+tomcat配置集群负载均衡

    在IT行业中,构建高效、可扩展的Web服务是至关重要的,而"**Nginx+Tomcat配置集群负载均衡**"就是实现这一目标的一种常见方案。Nginx是一款高性能的反向代理服务器,常用于处理静态资源和进行负载均衡;Tomcat则是...

    linux下配置tomcat集群的负载均衡

    总之,配置Linux下的Tomcat集群负载均衡是一项复杂但重要的工作,它能够显著提升系统的性能和稳定性,同时需要对集群管理、网络配置和服务器监控有深入的理解。通过合理的规划和实施,可以构建出一个强大且可靠的Web...

Global site tag (gtag.js) - Google Analytics