springboot输出流到页面_SpringBoot HttpServletResponse Header Cookie输出问题

本文探讨了在SpringBoot应用中遇到的Response写入Header和Cookie时部分信息丢失的问题。问题源于对Response的不当使用,尤其是误调用`flushBuffer()`导致响应提前提交。解决方案是避免在设置Header和Cookie时刷新输出流,确保所有操作在提交前完成。正确理解`isCommitted()`的状态至关重要,以防止头部信息被忽略。
摘要由CSDN通过智能技术生成

问题:

在一次Response写入header和cookie的时候,发现部分信息没有被输出

工具类:

CookieUtils:

import java.io.IOException;

import java.util.HashMap;

import java.util.Map;

import javax.annotation.PostConstruct;

import javax.servlet.http.Cookie;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import org.apache.commons.lang3.StringUtils;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.stereotype.Component;

/**

* Created by qhong on 2018/10/15 15:46

**/

@Component

public class CookieUtils {

public static final int COOKIE_MAX_AGE = 7 * 24 * 3600;

public static final int COOKIE_HALF_HOUR = 30 * 60;

private static HttpServletResponse response;

@Autowired

private HttpServletResponse response2;

private static HttpServletRequest request;

@Autowired

private HttpServletRequest request2;

@PostConstruct

public void beforeInit() {

request=request2;

response=response2;

}

/**

* 根据Cookie名称得到Cookie对象,不存在该对象则返回Null

*

* @param request

* @param name

* @return

*/

public static Cookie getCookie(String name) {

Cookie[] cookies = request.getCookies();

if (cookies==null||cookies.length<1) {

return null;

}

Cookie cookie = null;

for (Cookie c : cookies) {

if (name.equals(c.getName())) {

cookie = c;

break;

}

}

return cookie;

}

/**

* 根据Cookie名称直接得到Cookie值

*

* @param request

* @param name

* @return

*/

public static String getCookieValue(String name) {

Cookie cookie = getCookie(name);

if(cookie != null){

return cookie.getValue();

}

return null;

}

/**

* 移除cookie

* @param request

* @param response

* @param name 这个是名称,不是值

*/

public static void removeCookie(String name) {

if (null == name) {

return;

}

Cookie cookie = getCookie(name);

if(null != cookie){

cookie.setPath("/");

cookie.setValue("");

cookie.setMaxAge(0);

response.addCookie(cookie);

}

}

/**

* 添加一条新的Cookie,可以指定过期时间(单位:秒)

*

* @param response

* @param name

* @param value

* @param maxValue

*/

public static void setCookie(String name,

String value, int maxValue) {

if (StringUtils.isBlank(name)) {

return;

}

if (null == value) {

value = "";

}

Cookie cookie = new Cookie(name, value);

cookie.setPath("/");

if (maxValue != 0) {

cookie.setMaxAge(maxValue);

} else {

cookie.setMaxAge(COOKIE_HALF_HOUR);

}

response.addCookie(cookie);

// try {

// response.flushBuffer();

// } catch (IOException e) {

// e.printStackTrace();

// }

}

/**

* 添加一条新的Cookie,默认30分钟过期时间

*

* @param response

* @param name

* @param value

*/

public static void setCookie(String name,

String value) {

setCookie(name, value, COOKIE_HALF_HOUR);

}

/**

* 将cookie封装到Map里面

* @param request

* @return

*/

public static Map getCookieMap(){

Map cookieMap = new HashMap<>();

Cookie[] cookies = request.getCookies();

if(cookies!=null&&cookies.length>1){

for(Cookie cookie : cookies){

cookieMap.put(cookie.getName(), cookie);

}

}

return cookieMap;

}

}

SpringServletUtil:

import java.io.IOException;

import java.util.Map;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import lombok.extern.slf4j.Slf4j;

import org.apache.commons.lang3.StringUtils;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.stereotype.Component;

/**

* Created by qhong on 2019/1/11 14:17

**/

@Component

@Slf4j

public class SpringHttpUtil {

// /**

// * 获取请求体

// * @return

// */

// private HttpServletRequest getRequest(){

// return ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();

// }

//

// /**

// * 获取返回体

// * @return

// */

// private HttpServletResponse getResponse(){

// return ((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getResponse();

// }

@Autowired

private HttpServletResponse response;

@Autowired

private HttpServletRequest request;

/**

* 获取用户请求头部或者cookie中的参数

*/

public String getParams(String name) {

String result = getHeader(name);

//cookie

if (StringUtils.isBlank(result)) {

result = CookieUtils.getCookieValue(name);

}

return result;

}

public String getHeader(String name) {

String result= request.getHeader(name);

if(StringUtils.isBlank(result)){

result=request.getParameter(name);

}

return result;

}

public void setHeader(Map map) {

if (map == null || map.isEmpty() || map.size() == 0)

return;

map.entrySet().stream().forEach(x -> {

response.setHeader(x.getKey(), x.getValue());

});

}

public void setCookie(Map map) {

if (map == null || map.isEmpty() || map.size() == 0)

return;

map.entrySet().stream().forEach(x -> {

CookieUtils.setCookie(x.getKey(), x.getValue());

});

}

}

使用:

Map map = new HashMap<>();

map.put("aaaaa","aaa");

map.put("bbbbb","bbb");

springHttpUtil.setHeader(map);

springHttpUtil.setCookie(map);

很简单的测试使用 ,但是发现cookie只能输出一个

查看源码:

org\apache\tomcat\embed\tomcat-embed-core\8.5.15\tomcat-embed-core-8.5.15.jar

ResponseFacade:

@Override

public void addCookie(Cookie cookie) {

if (isCommitted()) {

return;

}

response.addCookie(cookie);

}

@Override

public void setHeader(String name, String value) {

if (isCommitted()) {

return;

}

response.setHeader(name, value);

}

@Override

public void addHeader(String name, String value) {

if (isCommitted()) {

return;

}

response.addHeader(name, value);

}

我也断点调试了,发现除了第一次isCommitted是false,其他的true,所以其他的参数才没有输出

刷新输出流

response内部的输出流有8KB的缓冲区,如果缓冲区满了的话,那么response会自动去提交,即把缓冲区内容输出给客户端。这时调用response的isCommited()方法返回的就是true,表示response已经提交过至少一次了。

也可以在缓冲区没有装满时调用response.flushBuffer()方法刷新输出流,把缓冲区中的数据发送到客户端去。同样,这也会导致response的isCommited()方法返回的就是true,表示response已经提交过至少一次了。

其实也可以调用response.getWirter().flush()方法达到与调用response.flushBuffer()相同的效果。这两种方式基本相同!

一旦response的isCommited()方法返回true,这说明服务器已经至少把状态码、响应头等数据发送给客户端了,也就是说已经开始向客户响应了。

错误原因:

错误的地方就是CookieUtils中被我注释掉的地方,这里对response进行了flushBuffer,所以isCommitted为true,后面的参数才会无效。

response.flushBuffer最好是只对返回主体内容使用,对于头部信息除非确定是最后了,否则不要使用,使用的话,后续对头部信息的任何操作都无效。

参考:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值