一. Nginx介绍.

Nginx是一款轻量级的Web 服务器/反向代理服务器及电子邮件其特点是占有内存少,并发能力强。

官网地址:
https://nginx.org/en/download.html

当一台tomcat无法承受客户端的压力时,对tomcat服务器搭建集群.
 Nginx让客户端通过一个统一的请求地址,去访问到tomcat服务器的集群.
 当发送大量请求时,Nginx可以合理的将全部的请求分发到tomcat服务器集群.
 Nginx中间件可以承受较大的并发量.

二. docker之Nginx安装.

1. 准备docker-compose.yml文件

version: '3.1'
services:
  nginx:
    restart: always
    image: daocloud.io/library/nginx:1.15.9
    container_name: nginx
    ports:
      - 80:80
    volumes: 
      - /opt/nginx-compose/conf.d/:/etc/nginx/conf.d/

2. Nginx的配置文件.

# 全局块
...              
# events块
events {         
   ...
}
# http块
http      
{
    # http全局块
    ...   
    # 虚拟主机server块
    server        
    { 
        # server全局块
        ...       
        # location块
        location [PATTERN]   
        {
            ...
        }
        location [PATTERN] 
        {
            ...
        }
    }
    server
    {
      ...
    }
    # http全局块
    ...     
}

全局块:配置影响nginx全局的指令。一般有运行nginx服务器的用户组,nginx进程pid存放路径,日志存放路径,配置文件引入,允许生成worker process数等

events块:配置影响nginx服务器或与用户的网络连接。有每个进程的最大连接数,选取哪种事件驱动模型处理连接请求,是否允许同时接受多个网路连接,开启多个网络连接序列化等

http块:可以嵌套多个server,配置代理,缓存,日志定义等绝大多数功能和第三方模块的配置。如文件引入,mime-type定义,日志自定义,是否使用sendfile传输文件,连接超时时间,单连接请求数等

server块:配置虚拟主机的相关参数,一个http中可以有多个server
location块:配置请求的路由,以及各种页面的处理情况

########### 每个指令必须有分号结束。#################
#user administrator administrators;  #配置用户或者组,默认为nobody nobody。
#worker_processes 2;  #允许生成的进程数,默认为1
#pid /nginx/pid/nginx.pid;   #指定nginx进程运行文件存放地址
error_log log/error.log debug;  #制定日志路径,级别。这个设置可以放入全局块,http块,server块,级别以此为:debug|info|notice|warn|error|crit|alert|emerg
events {
    accept_mutex on;   #设置网路连接序列化,防止惊群现象发生,默认为on
    multi_accept on;  #设置一个进程是否同时接受多个网络连接,默认为off
    #use epoll;      #事件驱动模型,select|poll|kqueue|epoll|resig|/dev/poll|eventport
    worker_connections  1024;    #最大连接数,默认为512
}
http {
    include       mime.types;   #文件扩展名与文件类型映射表
    default_type  application/octet-stream; #默认文件类型,默认为text/plain
    #access_log off; #取消服务日志    
    log_format myFormat '$remote_addr–$remote_user [$time_local] $request $status $body_bytes_sent $http_referer $http_user_agent $http_x_forwarded_for'; #自定义格式
    access_log log/access.log myFormat;  #combined为日志格式的默认值
    sendfile on;   #允许sendfile方式传输文件,默认为off,可以在http块,server块,location块。
    sendfile_max_chunk 100k;  #每个进程每次调用传输数量不能大于设定的值,默认为0,即不设上限。
    keepalive_timeout 65;  #连接超时时间,默认为75s,可以在http,server,location块。

    # 定义常量
    upstream mysvr {   
      server 127.0.0.1:7878;
      server 192.168.10.121:3333 backup;  #热备
    }
    error_page 404 https://www.baidu.com; #错误页 

    #定义某个负载均衡服务器   
    server {
        keepalive_requests 120; #单连接请求上限次数。
        listen       4545;   #监听端口
        server_name  127.0.0.1;   #监听地址       
        location  ~*^.+$ {       #请求的url过滤,正则匹配,~为区分大小写,~*为不区分大小写。
           #root path;  #根目录
           #index vv.txt;  #设置默认页
           proxy_pass  http://mysvr;  #请求转向mysvr 定义的服务器列表
           deny 127.0.0.1;  #拒绝的ip
           allow 172.18.5.54; #允许的ip           
        } 
    }
}

三. 反向代理.

1. 正向代理.

正向代理指的是一个位于客户端和原始服务器之间的服务器,为了从原始服务器取得内容,客户端向代理发送一个请求并指定目标(原始服务器),然后代理向原始服务器转交请求并将获得的内容返回给客户端

2. 反向代理.

反向代理(Reverse Proxy)方式是指以代理服务器来接受Internet上的连接请求,然后将请求转发给内部网络上的服务器,并将从服务器上得到的结果返回给Internet上请求连接的客户端,此时代理服务器对外就表现为一个服务器

3. 实现Nginx的反向代理配置.

3.1 准备一个tomcat服务器.

保证可以正常的访问tomcat服务.

3.2 准备Nginx并配置反向代理信息.

保证Nginx是可以正常访问的.
server{
	listen 80;						
	server_name localhost;			

	location / {					
		proxy_pass http://192.168.1.6:8080/;
	}
}

4. 实现通过路径访问不同的服务器.

4.1 准备两个tomcat服务器.

4.2 准备Nginx,并配置反向代理信息.

server{
	listen 80;						
	server_name localhost;	
    
    location /tomcat0/ {		#  http://192.168.1.6/tomcat0
    	proxy_pass http://192.168.1.6:8080/;
    }
    
    location /tomcat1/ {     #  http://192.168.1.6/tomcat1
    	proxy_pass http://192.168.1.6:8081/;
    }
    
}

四. 负载均衡.

1. 负载均衡策略.

1. 轮询,Nginx的默认机制,每个请求由Nginx逐个分发给不同的服务器.

2. 权重,由于每台tomcat的服务器配置不同,导致处理能力有偏差,平均分配不能将发挥配置较好的服务器,可以采用设置权重值的方式,给配置更好的服务器分配更多的请求.

3. ip_hash,根据发送请求的ip地址,来计算hash值,并最终决定将请求分发到哪台tomcat上,只要ip不变,当前客户端发起的请求,始终只会交给一台tomcat做处理.

2. 负载均衡-轮询.

2.1 准备tomcat集群.

2.2 准备Nginx,并配置负载均衡.

upstream my-server{  		# 名字中不要出现_
	server 192.168.1.6:8080;
	server 192.168.1.6:8081;
}

server{
	listen 80;						
	server_name localhost;	
    
    location / {
    	proxy_pass http://my-server/;		#  指定上了upstream中的 my-server
    }
    
}

3. 负载均衡-权重

# 权重只需要在轮询的基础上,给每一个tomcat节点添加weight参数即可.
upstream my-server{  		# 名字中不要出现_
	server 192.168.1.6 weight=10;
	server 192.168.1.6 weight=5;
}

server{
	listen 80;						
	server_name localhost;	
    
    location / {
    	proxy_pass http://my-server/;		#  指定上了upstream中的 my-server
    }
    
}

4. 负载均衡-ip_hash

upstream my-server{  		# 名字中不要出现 _
	ip_hash;
	server 192.168.1.6:8080;
	server 192.168.1.6:8081;
}

server{
	listen 80;						
	server_name localhost;	
    
    location / {
    	proxy_pass http://my-server/;		#  指定上了upstream中的 my-server
    } 
}

五. Nginx的动静分离.

1. Nginx的并发能力的计算.

Nginx的并发数计算:
	除以4是因为将请求转发到了动态的资源上.
		worker_processes × worker_connections / 4 = Nginx的最大并发数.
	如果你将请求转到了静态资源中,并发数的计算.
		worker_processes × worker_connections / 2 = Nginx的最大并发数.

2. 动态资源代理.

# 动态资源代理.
location / {
	proxy_pass 路径;
}

3. 静态资源代理.

3.1 准备静态资源.

1. 修改docker-compose.yml文件,添加两个数据卷,并将其映射到本地.
    volumes:
      - /opt/nginx-compose/conf.d/:/etc/nginx/conf.d/
      - /opt/nginx-compose/html/:/usr/share/nginx/html/
      - /opt/nginx-compose/img/:/usr/share/nginx/img/
      
2. 准备页面(html).
3. 准备图片.

3.2 准备Nginx,并添加静态资源代理配置.

server{
	listen 80;						
	server_name localhost;	
    
    location /html/ {					# http://192.168.1.6/html/
    	root /usr/share/nginx/;
    	index index.html;
    }
    
    location /img/ {					# http://192.168.1.6/img/
    	root /usr/share/nginx/;
    	autoindex on;					# 代表访问img时,展示当前目录下内容的全部列表
    }
}

六. Nginx解决跨域.

1. 跨域出现的原因&解决方案.

1. 浏览器开启了安全机制.(浏览器认为从另外一台服务器上返回的数据是不安全.)
2. 发送的请求必须是xhr请求.(ajax)   
3. 请求发起方和请求响应方的域不同的.(协议,ip,port)

2. 让浏览器关闭安全机制.

1. 找到了chrome浏览器的启动位置.

2. 打开了cmd窗口,并且使用下述命令,不开启安全机制的方式打开google浏览器.
	chrome.exe --disable-web-security --user-data-dir=D:\test-kuayu
	
不推荐使用,需要客户端|用户方面去解决跨域问题.

3. jsonp方式解决跨域.

3.1 jsonp解决跨域的应用

1. 只需要在发送ajax请求的位置,将dataType的值修改为jsonp即可.
2. 修改完毕之后,发送的ajax请求会改变为script请求.
	http://localhost/test/demo?
		callback=jQuery110208375363218521261_1581407642210&
		_=1581407642211
3. jsonp是一个解决跨域的约定,如果发送请求使用了jsonp,那么响应的数据必须放在以callback为名字的函数参数中.
	jQuery110208375363218521261_1581407642210({"code": 0,msg: "",data: "good!!!"})
	
4. 在ajax的success回到函数中,不需要有任何变化的.

3.2 jsonp解决跨域的问题.

1. jsonp只支持get请求方式.
2. 后台代码需要改变.
3. 不是xhr请求,xhr请求是由很多特性的.

4. 同源策略方式.

4.1 编写同源策略.

只需要在响应数据之前,添加指定的响应头信息即可.
按照火狐浏览器的提示: 
	已拦截跨源请求:同源策略禁止读取位于 http://localhost/test/demo 的远程资源。(原因:CORS 头缺少 'Access-Control-Allow-Origin')。
将响应头添加在了controler中 -> resp.setHeader("Access-Control-Allow-Origin","*");
如此操作既可以拿到跨域资源.

4.2 Spring也支持CORS处理跨域问题.

可以在Controller类上添加一个注解,这个注解会帮助我们自动添加相应头信息.
	@CrossOrigin
@CrossOrigin注解就是帮我们在响应数据时,添加相应的响应头信息.

6. Nginx处理跨域.

6.1 Nginx解决跨域的方式.

请求发起方和请求响应方的域不同的.

请求的响应方,添加相应头的方式处理.

# 访问demo.解决跨域问题
location / {
	proxy_pass http://192.168.1.6:8888/;
	# 解决跨域,添加响应头.
	add_header access-control-allow-origin '*';
}
请求的发起方,伪装域的方式处理.
伪装请求的信息,从协议,到ip,到port,都是一致的.
让一个nginx静态代理页面,并且同时动态代理controller接口即可.