博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
解决nginx负载均衡的session共享问题
阅读量:5954 次
发布时间:2019-06-19

本文共 3232 字,大约阅读时间需要 10 分钟。

查了一些资料,看了一些别人写的文档,总结如下,实现nginx session的共享

服务器有多台,用nginx做负载均衡,这样同一个IP访问同一个页面会被分配到不同的服务器上,如果session不同步的话,就会出现很多问题,比如说最常见的登录状态,

我们使用session无非是扩大变量的作用域,让我们在一次请求中可以随意获取session中存储的数据。详细信息查看:

下面提供了几种方式来解决session共享的问题:

1、不使用session,换用cookie

session是存放在服务器端的,cookie是存放在客户端的,我们可以把用户访问页面产生的session放到cookie里面,就是以cookie为中转站。你访问web服务器A,产生了session然后把它放到cookie里面,当你的请求被分配到B服务器时,服务器B先判断服务器有没有这个session,如果没有,再去看看客户端的cookie里面有没有这个session,如果也没有,说明session真的不存,如果cookie里面有,就把cookie里面的sessoin同步到服务器B,这样就可以实现session的同步了。

说明:这种方法实现起来简单,方便,也不会加大的负担,但是如果客户端把cookie禁掉了的话,那么session就无从同步了,这样会给网站带来损失;cookie的安全性不高,虽然它已经加了密,但是还是可以伪造的。

2、session存在数据库(等)中

php可以配置将session保存在数据库中,这种方法是把存放session的表和其他数据库表放在一起,如果mysql也做了集群了话,每个mysql节点都要有这张表,并且这张session表的数据表要实时同步。

说明:用数据库来同步session,会加库的IO,增加数据库的负担。而且数据库读写速度较慢,不利于session的适时同步。

3、session存在memcache或者中

memcache可以做分布式,php配置文件中设置存储方式为memcache,这样php自己会建立一个session集群,将session数据存储在memcache中。

说明:以这种方式来同步session,不会加大数据库的负担,并且安全性比用cookie大大的提高,把session放到内存里面,比从文件中读取要快很多。但是memcache把内存分成很多种规格的存储块,有块就有大小,这种方式也就决定了,memcache不能完全利用内存,会产生内存碎片,如果存储块不足,还会产生内存溢出。

4、nginx中的ip_hash技术能够将某个ip的请求定向到同一台后端,这样一来这个ip下的某个客户端和某个后端就能建立起稳固的session

ip_hash是在upstream配置中定义的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
upstream nginx.example.com    
    
{     
             
server 
192.168
.
74.235
:
80
;     
             
server 
192.168
.
74.236
:
80
;    
             
ip_hash;    
    
}    
    
server    
    
{    
             
listen 
80
;    
             
location /    
             
{    
                     
proxy_pass    
                    
http:
//nginx.example.com;    
             
}    
 
}

ip_hash是容易理解的,但是因为仅仅能用ip这个因子来分配后端,因此ip_hash是有缺陷的,不能在一些情况下使用:

a.nginx不是最前端的服务器。

ip_hash要求nginx一定是最前端的服务器,否则nginx得不到正确ip,就不能根据ip作hash。譬如使用的是squid为最前端,那么nginx取ip时只能得到squid的服务器ip地址,用这个地址来作分流是肯定错乱的。

b.nginx的后端还有其它方式的负载均衡。

假如nginx后端又有其它负载均衡,将请求又通过另外的方式分流了,那么某个客户端的请求肯定不能定位到同一台session应用服务器上。这么算起来,nginx后端只能直接指向应用服务器,或者再搭一个squid,然后指向应用服务器。最好的办法是用 location作一次分流,将需要session的部分请求通过ip_hash分流,剩下的走其它后端去。

c.被hash选中的服务器故障后,如何解决

5、upstream_hash

为了解决ip_hash的一些问题,可以使用upstream_hash这个第三方模块,这个模块多数情况下是用作url_hash的,但是并不妨碍将它用来做session共享。没试过真心的不明白

session存在memcache步骤:

            以下操作步骤的前提是你已经安装了,Tomcat和Memcached,和已经配置了Nginx的负载,

            如果这些您都没做,请先安装和配置,这些网上有很多,不再多说。

       1、下载jar包

                 下载的jar有:

                     memcached-session-manager-1.5.1.jar, memcached-session-manager-tc6-1.5.1.jar, 

                     jar包下载地址:

       2、把下载的jar包放到%TOMCAT_HOMT%\lib中。

       3、配置%TOMCAT_HOME%\config\context.xml

                在Context.xml文件中加入

1
2
3
4
5
6
7
8
9
<Manager className=
"de.javakaffee.web.msm.MemcachedBackupSessionManager"    
memcachedNodes=
"n1:localhost:11211,n2:localhost:11212"              
requestUriIgnorePattern=".*\.(png|gif|jpg|css|js){
1
}quot;    
sessionBackupAsync=
"false"    
sessionBackupTimeout=
"1800000"    
copyCollectionsForSerialization=
"false"    
transcoderFactoryClass=
"de.javakaffee.web.msm.serializer.javolution.JavolutionTranscoderFactory"    
              
     
/>

也可在server.xml配置文件的<Host>...<Host>中添加配置

1
2
3
4
5
6
7
8
9
10
11
<Context docBase=
"F:/tomcats/tomcat-7.0.2_2/webapps"  
path= 
"/msm"  
reloadable= 
"true"  
>    
      
<Manager className=
"de.javakaffee.web.msm.MemcachedBackupSessionManager"    
memcachedNodes=
"n1:localhost:11211,n2:localhost:11212"              
requestUriIgnorePattern=".*\.(png|gif|jpg|css|js){
1
}quot;    
sessionBackupAsync=
"false"    
sessionBackupTimeout=
"1800000"    
copyCollectionsForSerialization=
"false"    
transcoderFactoryClass=
"de.javakaffee.web.msm.serializer.javolution.JavolutionTranscoderFactory"    
              
/>    
</Context>
本文转自yunlielai51CTO博客,原文链接:http://blog.51cto.com/4925054/1909913
,如需转载请自行联系原作者
你可能感兴趣的文章
Spring学习总结(2)——Spring的常用注解
查看>>
关于IT行业人员吃的都是青春饭?[透彻讲解]
查看>>
钱到用时方恨少(随记)
查看>>
mybatis主键返回的实现
查看>>
org.openqa.selenium.StaleElementReferenceException
查看>>
Android Intent传递对象为什么要序列化?
查看>>
数论之 莫比乌斯函数
查看>>
linux下查找某个文件位置的方法
查看>>
python之MySQL学习——数据操作
查看>>
Harmonic Number (II)
查看>>
长连接、短连接、长轮询和WebSocket
查看>>
day30 模拟ssh远程执行命令
查看>>
做错的题目——给Array附加属性
查看>>
Url.Action取消字符转义
查看>>
JQuery选择器大全
查看>>
Gamma阶段第三次scrum meeting
查看>>
python3之装饰器修复技术@wraps
查看>>
[考试]20150606
查看>>
Javascript_备忘录5
查看>>
Can’t create handler inside thread that has not called Looper.prepare()
查看>>