Http知识汇总
前言
Github:https://github.com/HealerJean
1、Http协议
1.1、http连接
HTTP
协议即超文本传送协议(Hypertext Transfer Protocol
),是Web联网的基础,也是手机联网常用的协议之一,HTTP协议是建立在TCP协议之上的一种应用。
1)在HTTP 1.0(短连接)中,客户端的每次请求都要求建立一次单独的连接,在处理完本次请求后,就自动释放连接。
2)在HTTP 1.1(长连接)中则可以在一次连接中处理多个请求,并且多个请求可以重叠进行,不需要等待一个请求结束后再发送下一个请求。
由于HTTP在每次请求结束后都会主动释放连接,因此HTTP连接是一种“短连接”,为了让它变成长连接,要保持客户端程序的在线状态,需要不断地向服务器发起连接请求。通常的 做法是即时不需要获得任何数据,客户端也保持每隔一段固定的时间向服务器发送一次“保持连接”的请求,服务器在收到该请求后对客户端进行回复,表明知道客 户端“在线”。若服务器长时间无法收到客户端的请求,则认为客户端“下线”,若客户端长时间无法收到服务器的回复,则认为网络已经断开。
1.2、在浏览器中输入网址发生了什么
1、解析Web页面的URL,得到Web服务器的域名,查询本地的DNS缓存,当用户在浏览器中输入一个需要登录的网址时,系统会首先自动从Hosts文件中寻找对应的IP地址,以找到域名对应的主机IP地址(若有,则跳到4)查询远程域名根DNS,找到IP地址
2、通过DNS服务器获得Web服务器的IP地址
3、与Web服务器建立TCP连接
4、与Web服务器建立HTTP连接,向远程IP地址的服务器发送请求
5、从Web服务器获得URL指定的文档
6、浏览器解释页面文档,并显示在屏幕
1.3、TCP三次握手
所谓三次握手(Three-way Handshake),是指建立一个TCP连接时,需要客户端和服务器总共发送3个包
1.3.1、连接过程
SYN: (同步序列编号,Synchronize Sequence Numbers)
ACK: (确认编号,Acknowledgement Number)
FIN: (结束标志,FINish)
1.3.1.1、服务器和客户端初始状态 :CLOSED
:
1.3.1.2、服务器创建Socket后开始监听:LISTEN
;
1.3.1.4、第一次握手:客户端请求建立连接,客户端的状态:SYN_SENT
客户端请求建立连接,向服务器发送SYN报文,客户端的状态变成:
SYN_SENT
SYN=1,seq=x
SYN:置为1, 表示需要建立TCP连接
seq:序列号,是由发送端随机生成的
1.3.1.5、第二次握手:服务端发起,服务器状态:SYN_RECEIVED
服务器收到客户端的报文后向客户端发送ACK和SYN报文,此时服务器的状态变为
SYN_RCVD
;
SYN=1,ACK=x+1,seq=y
SYN :置为1
ACK: ACK字段数值是在客户端发送过来的序列号seq的基础上加1进行回复,以便客户端收到信息时,知晓自己的TCP建立请求已得到验证。ck加1可以理解为是确认和谁建立连接
seq: seq序列号,是由服务端端随机生成的
1.3.1.6、第三次握手:客户端发起,客户端和服务端状态:ESTABLISHED
客户端收到
ACK
、SYN
,就向服务器发送ACK
,客户端状态变为ESTABLISHED
,服务器端收到客户端的ACK后变为ESTABLISHED
。此时3次握手完成,连接建立!
SYN=1,ACK=y+1,seq=x+1
SYN :置为1
ACK: ACK字段数值是在服务端发送过来的序列号seq的基础上加1进行回复,以便服务端收到信息时,知晓自己的TCP建立请求已得到验证。
seq:客户端收到服务端发送的TCP建立验证请求后,会使自己的序列号加1表示,
1.3.2、状态解释
LISTEN:等待从任何远端TCP 和端口的连接请求。
SYN_SENT:发送完一个连接请求后等待一个匹配的连接请求。
SYN_RECEIVED:发送连接请求并且接收到匹配的连接请求以后等待连接请求确认。
ESTABLISHED:表示一个打开的连接,接收到的数据可以被投递给用户。连接的数据传输阶段的正常状态。
1.3.3、SYN攻击
1.3.3.1、攻击原理
1 、Client
伪造大量的虚假ip,向Server发送SYN包
2、Server
在接收到SYN
包后,会返回响应,并进入 SYN_RECV
状态,等待客户端的确认、但是伪造的ip肯定不会给予响应,于是Server
以为数据包丢失,不断重发,直到超时
1.3.3.2、危害
这些伪造的SYN
包会长期占用未连接队列,导致后来真实的ip无法加入队列,从而被丢弃,引起网络拥堵甚至网络瘫痪
1.3.3.3、如何确认是SYN攻击
当服务器上有大量的半连接且ip为随机的,可以确认是SYN攻击
netstat -nap | grep SYN_RECV
1.3.3.4、阻止
修改tcp协议实现。主要方法有SynAttackProtect保护机制、SYN cookies技术、增加最大半连接和缩短超时时间等
1.3.4、为什么是“三次握手?二次握手有什么问题?
为了防止已失效的连接请求报文段突然又传送到了服务端,因而产生错误”。
谢希仁版《计算机网络》中的例子是这样的,“已失效的连接请求报文段”的产生在这样一种情况下:
1、
client
发出的第一个连接请求报文段并没有丢失,而是在某个网络结点长时间的滞留了,以致延误到连接释放以后的某个时间才到达server
。2、本来这是一个早已失效的报文段。但
server
收到此失效的连接请求报文段后,就误认为是client
再次发出的一个新的连接请求。于是就向client
发出确认报文段,同意建立连接。3、假设不采用“三次握手”,那么只要
server
发出确认,新的连接就建立了。4、由于现在
client
并没有发出建立连接的请求,因此不会理睬server
的确认,也不会向server
发送数据。5、但
server
却以为新的运输连接已经建立,并一直等待client
发来数据。这样,server
的很多资源就白白浪费掉了。采用“三次握手”的办法可以防止上述现象发生。例如刚才那种情况,client
不会向server
的确认发出确认。server
由于收不到确认,就知道client
并没有要求建立连接。”如果采用了三次握手,由于客户端并没有发送新的连接请求,服务器端接收到的只是一个失效的报文段,所以客户端并不会理踩服务器端发送的确认,也不会向服务器端发送确认,此时服务器端就会知道客户端并没有请求建立连接,连接不会建立,也不会一直等待(超时断开)。
1.4、TCP四次挥手
1.4.1、握手过程
由于TCP连接是全双工的,断开连接会比建立连接麻烦一点点
1.4.1.1、第一次挥手 客户端发起,客户端状态为:FIN_WAIT1
客户端先向服务器发送FIN报文,请求断开连接,其状态变为。
FIN_WAIT1
;
FIN=1,seq=x
FIN:置为1
seq:发送端随机生成的
1.4.1.2、第二次挥手 服务端发起,服务端的状态:CLOSE_WAIT
服务端收到客户端请求断开的报文后,向客户端回复报文,此时服务器的状态变为:
CLOSE_WAIT
FIN=1 ACK=x+1 seq=y
FIN:置为1
ACK:字段,ACK字段数值是在客户端发过来的seq序列号基础上加1进行回复,以便客户端收到信息时,知晓自己的TCP断开请求已经得到验证。
seq:服务端随机生成
1.4.1.3、第三次挥手 服务端发起,服务端状态:LAST_ACK
服务端在回复完客户端的TCP断开请求后,不会马上进行TCP连接的断开,服务端会先确保断开前,所有传输到A的数据是否已经传输完毕,一旦确认传输数据完毕就会继续回复报文
FIN=1,ACK=x+1,seq=z
FIN:置为1
ACK:字段,ACK字段数值是在客户端发过来的seq序列号基础上加1进行回复,以便客户端收到信息时,知晓自己的TCP断开请求已经得到验证。
seq:服务端随机生成
1.4.1.4、第四次挥手 客户端的发起,等待收2次报文
客户端收到服务端第一次响应后变成·
FIN_WAIT2
状态,此时连接已经断开了一半了。如果服务器还有数据要发送给客户端,就会服务端确认完毕,客户端再次接受到报文的时候;客户端状态变成
TIME_WAIT状态
再过了2MSL长的时间后进入CLOSED状态。服务器收到客户端的ACK就进入CLOSED状态,
至此,还有一个状态没有出来:CLOSING状态:等待远端TCP 的连接终止请求确认。客户端发送了FIN,但是没有收到服务器的ACK,却收到了服务器的FIN,这种情况发生在服务器发送的ACK丢包的时候,因为网络传输有时会有意外。
FIN=1,ACK=z+1,seq=h
1.4.2、状态解释
FIN_WAIT_1:等待远端TCP 的连接终止请求,或者等待之前发送的连接终止请求的确认。
CLOSE_WAIT:等待本地用户的连接终止请求。
CLOSING:等待远端TCP 的连接终止请求确认。客户端发送了FIN,但是没有收到服务器的ACK,却收到了服务器的FIN,这种情况发生在服务器发送的ACK丢包的时候,因为网络传输有时会有意外。
LAST_ACK:等待先前发送给远端TCP 的连接终止请求的确认(包括它字节的连接终止请求的确认)
FIN_WAIT_2:等待远端TCP 的连接终止请求。
TIME_WAIT:等待足够的时间过去以确保远端TCP 接收到它的连接终止请求的确认。
TIME_WAIT 两个存在的理由:
1.可靠的实现tcp全双工连接的终止;
2.允许老的重复分节在网络中消逝。
CLOSED:不在连接状态(这是为方便描述假想的状态,实际不存在):
2、Request藏了什么东西
2.1、路径获取
2.1.1、request.getRequestURL
获取全部路径
log.info(request.getRequestURL().toString()); ////全部路径,不带参数 http://localhost:8081/healerjean/youhui/web/authorize
2.1.2、request.getContextPath()
2.2.3、request.getRequestURI()
//4、当然我们的Springboot项目中是没有用项目的名字的,下面来源于网络)
System.out.println(request.getContextPath()); ///可返回站点的根路径。也就是项目的名字 打印结果:/news
System.out.println(request.getServletPath()); //打印结果:/main/list.jsp
System.out.println(request.getRequestURI()); //打印结果:/news/main/list.jsp
我的项目中,因为没有项目的名字,所以下面二者显示的是一样的
log.info(request.getRequestURI()); //我的项目中没有项目的名字 healerjean/youhui/web/authorize
log.info(request.getServletPath()); //healerjean/youhui/web/authorize
2.2、request.getServerName
:获取服务器的 主机名
log.info(request.getServerName()); //localhost test.healerjean.cn
2.3、request.getServerPort()
:获取Htpp端口
log.info(request.getServerPort()+""); //8081 80
2.4、request.getLocalPort()
:获取本机端口
获取本机端口号(第二个是阿里云的)
log.info(request.getLocalPort()+""); //8081 8086
2.5、request.getHeader("host")
:
获取域名+端口(主机名加端口,80不显示)
log.info(request.getHeader("host")); //localhost:8081 test.healerjean.cn
2.6、request.getHeader( "referer" )
获取来源于上一次网页的url
如果是直接根据地址访问的则,则为null
log.info(request.getHeader( "referer" )); //进入这个地址之前是从那个连接过来的,防盗连接使用
2.7、request.getMethod
: 获取请求方法
request.getMethod() //POST 、GET
2.8、request.getContentType()
/ContentType
Content-Type是指http/https发送信息至服务器时的内容编码类型,contentType用于表明发送数据流的类型,服务器根据编码类型使用特定的解析方式,获取数据流中的数据。
1、GET和POST默认都是application/x-www-form-unlencoded
, 一般会指定,有时候不指定也可能遇到获取为null的情况
2、使用表单上传文件时,必须指定表单的 enctype属性值为 multipart/form-data. 请求体被分割成多部分,每部分使用 –boundary分割;
3、在网络请求中,常用的Content-Type有如下:
text/html,
text/plain,
text/css,
text/javascript,
image/jpeg,
image/png,
image/gif,
以上都是常见的页面资源类型。
application/x-www-form-urlencoded,
multipart/form-data,
application/json,
application/xml
这四个是ajax的请求,表单提交或上传文件的常用的资源类型。
2.9、request.getHeaderNames
获取请求头
2.9.1、遍历
request.getHeaderNames()
Enumeration headerNames=request.getHeaderNames();
while(headerNames.hasMoreElements()){
String paraName=(String)headerNames.nextElement();
System.out.println(paraName+": "+request.getHeader(paraName));
}
2.9.2、GET/POST返回的不同结果
2.9.2.1、GET
host: localhost:8081
connection: keep-alive
cache-control: max-age=0
upgrade-insecure-requests: 1
user-agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36
accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8
accept-encoding: gzip, deflate, br
accept-language: en,zh-CN;q=0.9,zh;q=0.8
cookie: _Token=9UUife5cUeoOLO4AOaExoA==; _aToken=ceshib; ab.storage.userId.8f0d3f99-1e3f-4444-bb6c-f73da32cc97c=%7B%22g%22%3A%22437277%22%2C%22c%22%3A1546077272288%2C%22l%22%3A1546077272288%7D; ab.storage.deviceId.8f0d3f99-1e3f-4444-bb6c-f73da32cc97c=%7B%22g%22%3A%229a4b0cd9-da59-9c0e-5aee-a8356e4ea5dd%22%2C%22c%22%3A1546077272296%2C%22l%22%3A1546077272296%7D; ab.storage.sessionId.8f0d3f99-1e3f-4444-bb6c-f73da32cc97c=%7B%22g%22%3A%22c137279a-3f33-6df2-b1fa-189f47793f93%22%2C%22e%22%3A1546079072315%2C%22c%22%3A1546077272293%2C%22l%22%3A1546077272315%7D; CONTAINERID=7485f72559e40cd7d5f09b1347a03d3a73e9a0be752d74c0f76412a132033784|XC3gl|XC3gW; JSESSIONID=13A2286D04353AD554B27FF77DF8CABA
2.9.2.1、POST
host: localhost:8081
connection: keep-alive
content-length: 453
user-agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36
cache-control: no-cache
origin: chrome-extension://fhbjgbiflinjbdggehcddcbncdddomop
postman-token: bfd0da34-bed9-b903-e852-5656de78883c
token: ZDjQ7mDEwlU7t6yEcAK2Xw==
content-type: multipart/form-data; boundary=----WebKitFormBoundaryaIFtRs1mZ4nS9I7j
accept: */*
accept-encoding: gzip, deflate, br
accept-language: en,zh-CN;q=0.9,zh;q=0.8
3.10、获取请求数据
3.10.1、request.getQueryString()
: 访问的参数
只能用于GET
log.info(request.getQueryString()); //参数 ,只适用于get
age=1&email=healerjewa%40gmail.com&name=healerjean&phone=17611115555
3.10.2、request.getParameterNames
不适用contentType为multipart/form-data
获取参数的名字,不管你代码里面接收不接收,反正我这里是可以获取到的
request.getParameterNames()
Enumeration enu=request.getParameterNames();
while(enu.hasMoreElements()){
String paraName=(String)enu.nextElement();
System.out.println(paraName+": "+request.getParameter(paraName));
}
参数如下:
accessDate: 2019-01-02 00:00:00
pageNum: 1
pageSize: 10
post: Healerjean
3.10.3、request.getParameterMap
获取参数map值: 不适用contentType为multipart/form-data
request.getParameterMap()
Map<String, String[]> parameterMap = request.getParameterMap();
log.info(JsonUtils.toJson(parameterMap));
for (Map.Entry<String, String[]> entry : parameterMap.entrySet()) {
System.out.println(entry.getKey());
for (String string : entry.getValue()) {
System.out.println(string);
}
System.out.println("----------------");
}
3.10.4、request.getParameter()
、request.getInputStream()
\ request.getReader()
3.10.4.1、getParameter(
)
get
和post
都可以使用,但如果是post
请求要根据表单提交数据的编码方式来确定能否使用.当编码方式是application/x- www-form-urlencoded
时才能使用(默认).这种编码方式虽然简单,但对于传输大块的二进制数据显得力不从心
3.10.5.2、getInputStream()
、getReader()
对于传输大块的二进制数这类数据,浏览器采用了另一种编码方式(“multipart/form-data”),这时就需要使用下面的两种方法.
getInputStream()
、getReader()
3.10.5.3、3中方式的使用冲突
3.10.5.3.1、冲突介绍
request.getParameter(
)、request.getInputStream()
、request.getReader()
这三种方法是有冲突的,因为流只能被读一次。
1、 当 form
表单内容采用 enctype=application/x-www-form-urlencoded
编码时,先通过调用request.getParameter()
方法得到参数后,再调用request.getInputStream()
或request.getReader()
已经得不到流中的内容,因为在调用 request.getParameter()时系统可能对表单中提交的数据以流的形式读了一次,反之亦然。
2、当 form
表单内容采用enctype=multipart/form-data
编码时,即使先调用request.getParameter()
也得不到数据,所以这时调用request.getParameter()
方法对 request.getInputStream()
或request.getReader()
没有冲突,即使已经调用了 request.getParameter()
方法也可以通过调用request.getInputStream()
或request.getReader()
得 到表单中的数据,
3、request.getInputStream()
和request.getReader()
在同一个响应中是不能混合使用的,如果混合使用就会抛异常。
3.10.5.3.2、让流可以被多次调用、在最后执行的过滤器中
package com.hlj.proj.config.filter;
import lombok.extern.slf4j.Slf4j;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
/**
* @author HealerJean
* @ClassName RequestFilter
* @date 2019-11-25 21:48.
* @Description
*/
@Slf4j
public class RequestFilter implements Filter {
private FilterConfig filterConfig;
@Override
public void init(FilterConfig filterConfig) throws ServletException {
this.filterConfig = filterConfig;
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws
IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
filterChain.doFilter(new ReuqestFiterHttpServletRequestWrapper(request), servletResponse);
}
@Override
public void destroy() {
this.filterConfig = null;
}
}
package com.hlj.proj.config.filter;
import org.apache.commons.io.IOUtils;
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.*;
public class ReuqestFiterHttpServletRequestWrapper extends HttpServletRequestWrapper {
private byte[] body;
/**
* 预先出初始化数据
*/
public ReuqestFiterHttpServletRequestWrapper(HttpServletRequest servletRequest) throws IOException {
super(servletRequest);
InputStream inputStream = servletRequest.getInputStream();
body = IOUtils.toByteArray(inputStream);
}
@Override
public BufferedReader getReader() throws IOException {
return new BufferedReader(new InputStreamReader(getInputStream()));
}
@Override
public ServletInputStream getInputStream() {
ByteArrayInputStream bis = new ByteArrayInputStream(body);
return new ServletInputStream() {
@Override
public int read() {
return bis.read();
}
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener readListener) {
}
};
}
}
3、Session和Cookie
3.1、跨域定位
URL | 说明 | 是否允许通信 |
---|---|---|
http://www.a.com/a.js http://www.a.com/b.js |
同一域名下 | 允许 |
http://www.a.com/lab/a.js http://www.a.com/script/b.js |
同一域名下不同文件夹 | 允许 |
http://www.a.com:8000/a.js http://www.a.com/b.js |
同一域名,不同端口 | 不允许 |
http://www.a.com/a.js https://www.a.com/b.js |
同一域名,不同协议 | 不允许 |
http://www.a.com/a.js http://70.32.92.74/b.js |
域名和域名对应ip | 不允许 |
http://www.a.com/a.js http://script.a.com/b.js |
主域相同,子域不同 | 不允许 |
http://www.a.com/a.js http://a.com/b.js |
同一域名,不同二级域名(同上) | 不允许(cookie这种情况下也不允许访问) |
http://www.cnblogs.com/a.js http://www.a.com/b.js |
不同域名 | 不允许 |
3.1、Session
3.1.1、问题汇总
3.1.1.1、 一个域名下项目session不能同时存在
之前做过一个Nginx,Tomcat,session共享,当时用了的是一个域名,也就是常见的localhost,当时没有考虑过一个域名下的session是不是同一个,但是端口不同,得到的正常情况下sessionId不是同一个。通过配置tomcat和项目,最后得到是一个sessionId。其实现在我才知道,如果没有配置共享session,本来一个localhost,不同端口,如果开启项目的话,每次刷新不同的端口项目,session变了。那是因为当前的端口将之前的端口的sessionId被替换掉了。但是之前的端口的sessionId再刷新的话,就又会变了。
接着说sso,我还是用了同一个localhost,发现,单点是可以登录,但是只能登录一个客户端。这就偶是3中强调的一个域名session是同一个,如果登录另外一个。通过sso就会将之前的session给干掉。只留下当前登录的那个客户端。我这里用的是sso使用的是ticket验证,不可以共用一个sessionId的。所以请看下面的第一个
3.1.2、session存储和使用
存储在服务器的内存中,tomcat的StandardManager类将session存储在内存中,也可以持久化到file,数据库,memcache,redis等。 session销毁只能通过invalidate或超时,关掉浏览器并不会关闭session。 Session的默认失效时间是30分钟,
3.1.3、 从session中获取数据
HttpSession session = request.getSession();
System.out.println(session.getId());
UserDTO user = session.getAttribute("USER"):
3.1.4、遍历Session
//获取session
HttpSession session = request.getSession();
// 获取session中所有的键值
Enumeration<String> attrs = session.getAttributeNames();
// 遍历attrs中的
while(attrs.hasMoreElements()){
// 获取session键值
String name = attrs.nextElement().toString();
// 根据键值取session中的值
Object vakue = session.getAttribute(name);
// 打印结果
System.out.println("------" + name + ":" + vakue +"--------\n");
}
sessionName::_const_cas_assertion_-value:org.jasig.cas.client.validation.AssertionImpl@12d08532
3.4、Cookie
3.4.1、遍历Cookie
Cookie[] cookies = request.getCookies();
String JSESSIONID = null;
String loginSessionId = null ;
for(Cookie cookie:cookies){
cookie.getValue() ;
}
3.5、结合单点登录看
看另一篇文章,单点登录原理
工具类
1、Ip工具类
package com.hlj.utils.ip;
import com.hlj.utils.ExceptionLogUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
import java.net.InetAddress;
import java.net.UnknownHostException;
/**
* @Desc:
* @Author HealerJean
* @Date 2018/7/4 下午2:36.
*/
public class IpUtil {
public static String getIp(){
HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
if(request==null)return null;
String ip = request.getHeader("x-forwarded-for");
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("X-Real-IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getRemoteAddr();
}
if(ip.toLowerCase().contains("x-forwarded-for")){
String ip_temp = "";
if(ip.contains(":") && ip.split(":").length>0 && !ip.endsWith(":")){
ip_temp = ip.split(":")[1];
if(StringUtils.isBlank(ip_temp)) ip_temp = ip.substring(0,ip.lastIndexOf("X-Forwarded-For"));
}else{
ip_temp = ip.substring(0,ip.lastIndexOf("X-Forwarded-For"));
}
ip = ip_temp.replaceAll(" ","");
}
if (ip.indexOf(",") > -1) {
String ip_temp = ip.split(",")[0];
ip_temp = ip_temp.replaceAll(" ", "");
if(ip_temp.startsWith("10.") && ip.split(",").length>1){
ip = ip.split(",")[1];
ip = ip.replaceAll(" ", "");
}else ip = ip_temp;
}
return ip;
}
/**
* 获取调用的http/https+ 域名 + 端口
* @param urlTarget
* @return
*/
public static String getDomainAndPort(String urlTarget)
{
//跳转到对应的回调地址
String domain = "";
try {
URL url = new URL(urlTarget);
String host = url.getHost();
int port = url.getPort();
String s = url.toString();
domain = s.substring(0,s.indexOf(host)+host.length());
if(port != -1) {
domain = domain + ":" + port;
}
} catch (MalformedURLException e) {
log.info("获取域名失败");
}
return domain ;
}
/**
* 获取服务器ip
* @return
*/
public static String getHostIp(){
try {
InetAddress ia2 = InetAddress.getLocalHost();
return ia2.getHostAddress();
} catch (UnknownHostException e) {
ExceptionLogUtils.log(e,IpUtil.class);
return null ;
}
}
/**
* 根据域名获取ip
* www.baidu.com
* @param url
* @return
*/
public static String getIpByUrl(String url) {
try {
InetAddress ia2 = InetAddress.getByName(url);
return ia2.getHostAddress();
} catch (UnknownHostException e) {
ExceptionLogUtils.log(e,IpUtil.class );
return null;
}
}
public static IPEntry getAddress(String ip){
return IPSeeker.getInstance().getAddress(ip);
}
public static void main(String[] args) {
System.out.println(getAddress("106.39.75.134"));
}
}
2、Request 工具类
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
/**
* 静态获取HttpServletRequest 和 session的方法
* 要使用此类需要在web.xml注册org.springframework.web.context.request.RequestContextListener
*/
public class RequestHolder {
public static HttpSession getSession() {
HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
return request.getSession();
}
public static HttpServletRequest getRequest() {
return ((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getRequest();
}
public static HttpServletResponse getResponse() {
return ((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getResponse();
}
}
3、Cookie工具类
package com.duodian.youhui.admin.utils.cookie;
import com.duodian.youhui.admin.utils.RequestHolder;
import org.apache.commons.lang3.StringUtils;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* @Desc: Cookie 工具类
* @Date: 2018/9/12 下午1:02.
*/
public class CookieHelper {
public static void setCookie(String key,String value,Integer seconds){
HttpServletResponse response = RequestHolder.getResponse();
Cookie add = new Cookie(key,value);
add.setMaxAge(seconds);
add.setPath("/");
response.addCookie(add);
}
public static String getCookieValue(String key){
HttpServletRequest request = RequestHolder.getRequest();
Cookie[] cookies = request.getCookies();
if (cookies == null){
return null;
}
for (Cookie cookie : cookies){
if (StringUtils.equals(cookie.getName(), key)){
return cookie.getValue();
}
}
return null;
}
public static void clearCookie(String key){
HttpServletResponse response = RequestHolder.getResponse();
Cookie clear = new Cookie(key,"");
clear.setMaxAge(0);
clear.setComment("清除cookie");
clear.setPath("/");
response.addCookie(clear);
}
}
3、Session工具
package com.duodian.admore.home.utils;
import com.duodian.admore.constants.AppConstants;
import com.duodian.admore.core.spring.RequestHolder;
import com.duodian.admore.entity.db.user.User;
import javax.servlet.http.HttpSession;
public class SessionUtils {
public static final String SESSION_USER = "user";
public static void initSession(User user){
HttpSession session = RequestHolder.getSession();
session.setAttribute(SESSION_USER,user);
}
public static void clearSession(){
HttpSession session = RequestHolder.getSession();
session.invalidate();
}
public static User getSessionUser(){
HttpSession session = RequestHolder.getSession();
User user = (User)session.getAttribute(SESSION_USER);
return user;
}
}
4、收集URI
import com.sankuai.windmill.riding.core.api.utils.URIUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Service;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.mvc.method.RequestMappingInfo;
import org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping;
import java.util.Map;
import java.util.Set;
@Slf4j
@Service
public class UriCollectRunner implements ApplicationRunner {
@Autowired
private WebApplicationContext applicationContext;
/**
* 获取所有的uri
*/
@Override
public void run(ApplicationArguments args) {
RequestMappingHandlerMapping mapping = applicationContext.getBean(RequestMappingHandlerMapping.class);
Map<RequestMappingInfo, HandlerMethod> map = mapping.getHandlerMethods();
for (RequestMappingInfo info : map.keySet()) {
Set<String> patterns = info.getPatternsCondition().getPatterns();
for (String uri : patterns) {
URIUtils.PROJECT_URI.add(uri);
}
}
}
}