1、web-server的负载均衡nginx
互联网架构中,web-server接入通常使用nginx来作反向代理,实施负载均衡。整个架构分三层:web
上游调用层,通常是browser或者APPapache
中间反向代理层,nginxtomcat
下游真实接入集群,web-server,常见web-server的有tomcat,apache服务器
整个访问过程为:架构
browser向daojia.com发起请求负载均衡
DNS服务器将daojia.com解析为外网IP(1.2.3.4)tcp
browser经过外网IP(1.2.3.4)访问nginx性能
nginx实施负载均衡策略,常见策略有轮询,随机,IP-hash等设计
nginx将请求转发给内网IP(192.168.0.1)的web-server
因为http短链接,以及web应用无状态的特性,理论上任何一个http请求落在任意一台web-server都应该获得正常处理(若是必须落在一台,说明架构不合理,不能水平扩展)。
问题来了,tcp是有状态的链接,客户端和服务端一旦创建链接,一个client发起的请求必须落在同一台tcp-server上,此时如何作负载均衡,如何保证水平扩展呢?
2、单机法tcp-server
单个tcp-server显然是能够保证请求一致性:
client向tcp.daojia.com发起tcp请求
DNS服务器将tcp.daojia.com解析为外网IP(1.2.3.4)
client经过外网IP(1.2.3.4)向tcp-server发起请求
方案的缺点?
没法保证高可用。
3、集群法tcp-server
经过搭建tcp-server集群来保证高可用,客户端来实现负载均衡:
client内配置有tcp1/tcp2/tcp3.daojia.com三个tcp-server的外网IP
客户端经过“随机”的方式选择tcp-server,假设选择到的是tcp1.daojia.com
经过DNS解析tcp1.daojia.com
经过外网IP链接真实的tcp-server
如何保证高可用呢?
若是client发现某个tcp-server链接不上,则选择另外一个。
潜在的缺点?
每次链接前,须要多实施一次DNS访问:
难以预防DNS劫持
多一次DNS访问意味着更长的链接时间,这个不足在手机端更为明显
如何解决DNS的问题?
直接将IP配置在客户端,能够解决上述两个问题,不少公司也就是这么作的(俗称“IP直通车”)。
“IP直通车”有什么新问题?
将IP写死在客户端,在客户端实施负载均衡,扩展性不好:
若是原有IP发生变化,客户端得不到实时通知
若是新增IP,即tcp-sever扩容,客户端也得不到实时通知
若是负载均衡策略变化,须要升级客户端
4、服务端实施负载均衡
只有将复杂的策略下沉到服务端,才能根本上解决扩展性的问题。
增长一个http接口,将客户端的“IP配置”与“均衡策略”放到服务端是一个不错的方案:
client每次访问tcp-server前,先调用一个新增的get-tcp-ip接口,对于client而言,这个http接口只返回一个tcp-server的IP
这个http接口,实现的是原client的IP均衡策略
拿到tcp-server的IP后,和原来同样向tcp-server发起TCP长链接
这样的话,扩展性问题就解决了:
若是原有IP发生变化,只须要修改get-tcp-ip接口的配置
若是新增IP,也是修改get-tcp-ip接口的配置
若是负载均衡策略变化,须要升级客户端
然而,新的问题又产生了,若是全部IP放在客户端,当有一个IP挂掉的时候,client能够再换一个IP链接,保证可用性,而get-tcp-ip接口只是维护静态的tcp-server集群IP,对于这些IP对应的tcp-server是否可用,是彻底不知情的,怎么办呢?
5、tcp-server状态上报
get-tcp-ip接口怎么知道tcp-server集群中各台服务器是否可用呢,tcp-server主动上报是一个潜在方案,若是某一个tcp-server挂了,则会终止上报,对于中止上报状态的tcp-server,get-tcp-ip接口,将不返回给client相应的tcp-server的外网IP。
该设计的存在的问题?
诚然,状态上报解决了tcp-server高可用的问题,但这个设计犯了一个“反向依赖”的耦合小错误:使得tcp-server要依赖于一个与自己业务无关的web-server。
6、tcp-server状态拉取
更优的方案是:web-server经过“拉”的方式获取各个tcp-server的状态,而不是tcp-server经过“推”的方式上报本身的状态。
这样的话,每一个tcp-server都独立与解耦,只需专一于资深的tcp业务功能便可。
高可用、负载均衡、扩展性等任务由get-tcp-ip的web-server专一来执行。
多说一句,将负载均衡实如今服务端,还有一个好处,能够实现异构tcp-server的负载均衡,以及过载保护:
静态实施:web-server下的多个tcp-server的IP能够配置负载权重,根据tcp-server的机器配置分配负载(nginx也有相似的功能)
动态实施:web-server能够根据“拉”回来的tcp-server的状态,动态分配负载,并在tcp-server性能极具降低时实施过载保护
7、总结
web-server如何实施负载均衡?
利用nginx反向代理来轮询、随机、ip-hash。
tcp-server怎么快速保证请求一致性?
单机。
如何保证高可用?
客户配置多个tcp-server的域名。
如何防止DNS劫持,以及加速?
IP直通车,客户端配置多个tcp-server的IP。
如何保证扩展性?
服务端提供get-tcp-ip接口,向client屏屏蔽负载均衡策略,并实施便捷扩容。
如何保证高可用?
tcp-server“推”状态给get-tcp-ip接口,
or
get-tcp-ip接口“拉”tcp-server状态。