一、重定向和转发
1.1 重定向和转发的区别
- 重定向后URL会变,转发URL不会变
- 重定向后台做了两次请求,转发一共就一次请求
- 重定向的地址在传统ServletAPI的方式里可不带“/”,转发的地址要带”/“
- 重定向可访问外网或者本web下的页面,转发只能是本web下的页面
- 重定向不可以走WEB-INF下的目录,因为WEB-INF是安全目录,客户端无法访问,而重定向就相当于用户直接从客户端访问了的路径;转发可以访问到WEB-INF下的页面
1.2 传统方式:ServletAPI
1.通过HttpServletResponse实现重定向
@Controller
public class ResultGo {
@RequestMapping("/go1")
public void test1(HttpServletRequest request, HttpServletResponse response) throws IOException {
//重定向,url直接改变
response.sendRedirect("index.jsp");
}
2.通过HttpServletRequest实现转发
@Controller
public class ResultGo {
@RequestMapping("/go2")
public void test2(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
//转发
request.getRequestDispatcher("/WEB-INF/jsp/test.jsp").forward(request,response);
}
}
1.3 springMVC方式
1.无视图解析器
重定向和转发
@Controller
public class ResultSpringMVC {
@RequestMapping("/springMVC-redirect")
public String test1(){
//重定向
return "redirect:index.jsp";
}
@RequestMapping("/springMVC-forward")
public String test2(Model model){
//转发
model.addAttribute("msg","springMVC-forward转发");
return "forward:/WEB-INF/jsp/test.jsp";
}
}
2.有视图解析器
视图解析器:页面={前缀}+名称+{后缀}
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
<!-- 前缀 -->
<property name="prefix" value="/WEB-INF/jsp/"/>
<!-- 后缀 -->
<property name="suffix" value=".jsp"/>
</bean>
重定向和转发
@Controller
public class ResultSpringMVC {
@RequestMapping("/redirect/viewResolve")
public String test3(){
//重定向,有视图解析器
return "redirect:/index.jsp";
}
@RequestMapping("/forward/viewResolve")
public String test4(Model model){
//转发,无视图解析器
model.addAttribute("msg","springMVC-forward转发");
return "test";
}
}
注意点:重定向与有无视图解析器无关的,但需要注意的是在传统的方式里地址可以省略开头带的“/”,但在有视图解析器后,若少了“/”,就会被认为是“redirect:index.jsp”某个页面名称,然后经过解析之后,就会出错。此外,或许重定向的url下会有波浪线,表示无法解析,直接无视就可以了,因为重定向根本不走解析。
二、中文乱码问题
方法一:spring提供一个过滤器
<filter>
<filter-name>encoding</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>utf-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>encoding</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
方法二:修改tomact配置文件(一般在安装时候就会修改)
<Connector URIEncoding="utf-8" port="8080" protocol="HTTP/1.1"
connectionTimeout="20000"
redirectPort="8443" />
方法三:自定义一个过滤器(大神写的,记得配置这个过滤器)
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Map;
/**
* 解决get和post请求 全部乱码的过滤器
*/
public class GenericEncodingFilter implements Filter {
@Override
public void destroy() {
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
//处理response的字符编码
HttpServletResponse myResponse=(HttpServletResponse) response;
myResponse.setContentType("text/html;charset=UTF-8");
// 转型为与协议相关对象
HttpServletRequest httpServletRequest = (HttpServletRequest) request;
// 对request包装增强
HttpServletRequest myrequest = new MyRequest(httpServletRequest);
chain.doFilter(myrequest, response);
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
}
//自定义request对象,HttpServletRequest的包装类
class MyRequest extends HttpServletRequestWrapper {
private HttpServletRequest request;
//是否编码的标记
private boolean hasEncode;
//定义一个可以传入HttpServletRequest对象的构造函数,以便对其进行装饰
public MyRequest(HttpServletRequest request) {
super(request);// super必须写
this.request = request;
}
// 对需要增强方法 进行覆盖
@Override
public Map getParameterMap() {
// 先获得请求方式
String method = request.getMethod();
if (method.equalsIgnoreCase("post")) {
// post请求
try {
// 处理post乱码
request.setCharacterEncoding("utf-8");
return request.getParameterMap();
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
} else if (method.equalsIgnoreCase("get")) {
// get请求
Map<String, String[]> parameterMap = request.getParameterMap();
if (!hasEncode) { // 确保get手动编码逻辑只运行一次
for (String parameterName : parameterMap.keySet()) {
String[] values = parameterMap.get(parameterName);
if (values != null) {
for (int i = 0; i < values.length; i++) {
try {
// 处理get乱码
values[i] = new String(values[i]
.getBytes("ISO-8859-1"), "utf-8");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
}
}
}
hasEncode = true;
}
return parameterMap;
}
return super.getParameterMap();
}
//取一个值
@Override
public String getParameter(String name) {
Map<String, String[]> parameterMap = getParameterMap();
String[] values = parameterMap.get(name);
if (values == null) {
return null;
}
return values[0]; // 取回参数的第一个值
}
//取所有值
@Override
public String[] getParameterValues(String name) {
Map<String, String[]> parameterMap = getParameterMap();
String[] values = parameterMap.get(name);
return values;
}
}