Spring MVC框架:第三章:获取请求参数和页面跳转控制

简介: Spring MVC框架:第三章:获取请求参数和页面跳转控制

获取请求参数

1.什么是请求参数?

<a href="emp/remove?empId=3">删除</a>
<form action="emp/save" method="post">
    姓名:<input type="text" name="empName"/><br/>
    年龄:<input type="text" name="empAge"/><br/>
    工资:<input type="text" name="empSalary"/><br/>
    <input type="submit" value="保存"/>
</form>

2.请求参数的四种情况

①一名一值

<a href=“emp/remove?empId=3”>删除</a>

在handler方法的参数上使用@RequestParam注解。

@RequestMapping("/caseOne")
public String caseOne(@RequestParam("empId") Integer empId) {
    System.out.println("empId="+empId);
    return "result";
}

SpringMVC会自动帮我们进行类型转换。如果请求参数的名字和handler方法中对应形参的名字一致那么可以省略@RequestParam注解。

@RequestMapping("/caseOne")
public String caseOne(Integer empId) {
    System.out.println("empId="+empId);
    return "result";
}

②一名多值

<form action="team" method="post">
    请选择你最喜欢的球队:
    <input type="checkbox" name="team" value="Brazil"/>巴西
    <input type="checkbox" name="team" value="German"/>德国
    <input type="checkbox" name="team" value="French"/>法国
    <input type="checkbox" name="team" value="Holland"/>荷兰
    <input type="checkbox" name="team" value="Italian"/>意大利
    <input type="checkbox" name="team" value="China"/>中国
    <br/>
    <input type="submit" value="保存"/>
</form>

使用List或数组来接收。

@RequestMapping("/caseTwo")
public String caseTwo(@RequestParam("team") List<String> teams) {
    System.out.println(teams);
    return "result";
}
@RequestMapping("/caseTwo")
public String caseTwo(@RequestParam("team") String[] teams) {
    System.out.println(Arrays.asList(teams));
    return "result";
}

③表单数据正好对应一个实体类

<form action="emp/save" method="post">
    姓名:<input type="text" name="empName"/><br/>
    年龄:<input type="text" name="empAge"/><br/>
    工资:<input type="text" name="empSalary"/><br/>
    <input type="submit" value="保存"/>
</form>

实体类:

public class Employee {
    private Integer empId;
    private String empName;
    private int empAge;
    private double empSalary;
    ……

直接使用和表单对应的实体类类型接收

@RequestMapping("/caseThree")
public String caseThree(Employee employee) {
    System.out.println(employee);
    return "result";
}

④表单对应的实体类包含级联属性

public class Student {
  private Integer studentId;
  private String studentName;
  private School school;
  private List<Subject> subjectList;
  private Subject[] subjectArray;    
  private Set<Teacher> teacherSet;      
  private Map<String,String> scoreMap;
  {
   //在各种常用数据类型中,只有Set类型需要提前初始化
        //并且要按照表单将要提交的对象数量进行初始化
        //Set类型使用非常不便,要尽可能避免使用Set
    teacherSet = new HashSet<>();
    teacherSet.add(new Teacher());
    teacherSet.add(new Teacher());
    teacherSet.add(new Teacher());
  }

handler方法

@RequestMapping("/get/param/multi/value")
  public String getParamOneNameMultiValue(@RequestParam("team") List<String> teamList) {        
    for (String team : teamList) {
      System.out.println(team);
    }       
    return "target";
  }
  @RequestMapping("/get/param/entity")
  public String getParamEntity(Employee employee) {       
    System.out.println(employee);       
    return "target";
  }
  //@RequestMapping("/get/param/entity")
  public String getParamEntityParam(@RequestParam("empName") String empName) {        
    System.out.println("empName="+empName);       
    return "target";
  }
  @RequestMapping("/get/param/fuza")
  public String getParamFuza(Student student) {
    System.out.println("StudentId="+student.getStudentId());
    System.out.println("StudentName="+student.getStudentName());
    System.out.println("SchoolId="+student.getSchool().getSchoolId());
    System.out.println("SchoolName="+student.getSchool().getSchoolName());
    List<Subject> subjectList = student.getSubjectList();
    for (Subject subject : subjectList) {
      System.out.println("科目名称="+subject.getSubjectName());
    }
    Subject[] subjectArray = student.getSubjectArray();
    for (Subject subject : subjectArray) {
      System.out.println("科目名称="+subject.getSubjectName());
    }
    Set<Teacher> teacherSet = student.getTeacherSet();
    for (Teacher teacher : teacherSet) {
      System.out.println("老师姓名="+teacher.getTeacherName());
    }
    Map<String, String> scoreMap = student.getScoreMap();
    Set<String> keySet = scoreMap.keySet();
    for (String key : keySet) {
      String value = scoreMap.get(key);
      System.out.println(key+":"+value);
    }
    return "target";
  }

提交数据的表单

<form action="${pageContext.request.contextPath }/get/param/fuza" method="post">
    学生编号:<input type="text" name="studentId" value="22" /><br/>
    学生姓名:<input type="text" name="studentName" value="tom" /><br/>
    <!-- getSchool().setSchoolId() -->
    学校编号:<input type="text" name="school.schoolId" value="33" /><br/>
    学校名称:<input type="text" name="school.schoolName" value="at" /><br/>
    科目1名称:<input type="text" name="subjectList[0].subjectName" value="理论" /><br/>
    科目2名称:<input type="text" name="subjectList[1].subjectName" value="倒库"/><br/>
    科目3名称:<input type="text" name="subjectList[2].subjectName" value="路考"/><br/>
    科目4名称:<input type="text" name="subjectArray[0].subjectName" value="撞人"/><br/>
    科目5名称:<input type="text" name="subjectArray[1].subjectName" value="防碰瓷"/><br/>
    科目6名称:<input type="text" name="subjectArray[2].subjectName" value="追尾"/><br/>
    老师姓名:<input type="text" name="teacherSet[0].teacherName" value="卡卡西"/><br/>
    老师姓名:<input type="text" name="teacherSet[1].teacherName" value="伊鲁卡"/><br/>
    老师姓名:<input type="text" name="teacherSet[2].teacherName" value="大蛇丸"/><br/>
    考试成绩:<input type="text" name="scoreMap['shuxue']" value="25" /><br/>
    考试成绩:<input type="text" name="scoreMap['yuwen']" value="16" /><br/>
    考试成绩:<input type="text" name="scoreMap['yingyu']" value="7" /><br/>
    考试成绩:<input type="text" name="scoreMap['lol']" value="100" /><br/>
    考试成绩:<input type="text" name="scoreMap['dota']" value="300" /><br/>
    <input type="submit" value="提交"/>
  </form>
  <br/><br/>
  <form action="${pageContext.request.contextPath }/get/param/entity" method="post">
    <!-- private Integer empId; -->
    编号:<input type="text" name="empId"/><br/>
    <!-- private String empName; -->
    姓名:<input type="text" name="empName"/><br/>
    <!-- private Integer empAge; -->
    年龄:<input type="text" name="empAge"/><br/>
    <!-- private Double empSalary; -->
    工资:<input type="text" name="empSalary"/><br/>
    <input type="submit" value="提交"/>
  </form>
  <br/><br/>
  <form action="${pageContext.request.contextPath }/get/param/multi/value" method="post">
    请选择你最喜欢的球队:<br/>
    <input type="checkbox" name="team" value="German"/>德国<br/>
    <input type="checkbox" name="team" value="Brazil"/>巴西<br/>
    <input type="checkbox" name="team" value="Italian"/>意大利<br/>
    <input type="checkbox" name="team" value="China"/>中国<br/>
    <input type="checkbox" name="team" value="Korea"/>韩国<br/>
    <input type="submit" value="提交"/>
  </form>

web.xml配置,解决字符乱码

<!-- 在SpringMVC环境下对请求和响应过程强制使用UTF-8字符集进行编码、解码 -->
  <filter>
    <filter-name>CharacterEncodingFilter</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>
    <init-param>
      <param-name>forceEncoding</param-name>
      <param-value>true</param-value>
    </init-param>
  </filter>
  <filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

页面跳转控制

1.转发指令

@RequestMapping("/testForward")
public String testForward() {
    return "forward:/target.jsp";
}

forward:/target.jsp

表示方法执行完成后转发到/target.jsp。想一想有了视图解析器拼接字符串指定转发页面之后forward是否多余呢?

2.重定向指令

@RequestMapping("/testRedirect")
public String testRedirect() {
    return "redirect:/target.jsp";
}

redirect:/target.jsp

表示方法执行完成后重定向到/target.jsp。这里重定向中使用的路径和我们以前的写法不同,没有以Web应用名称开头,这是因为SpringMVC会替我们加上。

3.使用原生对象完成转发

@RequestMapping("/testForward")
public void testForward2(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    request.getRequestDispatcher("/target.jsp").forward(request, response);
}

你看,使用原生request对象执行转发后,handler方法的返回值就必须是void,意思是我们自己指定了响应方式,不需要SpringMVC再进行处理了。一个请求只能有一个响应,不能在handler方法里面给一个,然后SpringMVC框架再给一个。

4.使用原生对象完成重定向

@RequestMapping("/testRedirect")
public void testRedirect2(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    response.sendRedirect(request.getContextPath()+"/target.jsp");
}

使用原生response对象执行重定向后,handler方法的返回值同样需要设置为void,原因同上。

相关文章
|
1月前
|
XML 安全 Java
|
2月前
|
缓存 NoSQL Java
什么是缓存?如何在 Spring Boot 中使用缓存框架
什么是缓存?如何在 Spring Boot 中使用缓存框架
53 0
|
9天前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
4天前
|
Java 开发者 Spring
理解和解决Spring框架中的事务自调用问题
事务自调用问题是由于 Spring AOP 代理机制引起的,当方法在同一个类内部自调用时,事务注解将失效。通过使用代理对象调用、将事务逻辑分离到不同类中或使用 AspectJ 模式,可以有效解决这一问题。理解和解决这一问题,对于保证 Spring 应用中的事务管理正确性至关重要。掌握这些技巧,可以提高开发效率和代码的健壮性。
28 13
|
4天前
|
监控 前端开发 API
一款基于 .NET MVC 框架开发、功能全面的MES系统
一款基于 .NET MVC 框架开发、功能全面的MES系统
|
16天前
|
IDE Java 测试技术
互联网应用主流框架整合之Spring Boot开发
通过本文的介绍,我们详细探讨了Spring Boot开发的核心概念和实践方法,包括项目结构、数据访问层、服务层、控制层、配置管理、单元测试以及部署与运行。Spring Boot通过简化配置和强大的生态系统,使得互联网应用的开发更加高效和可靠。希望本文能够帮助开发者快速掌握Spring Boot,并在实际项目中灵活应用。
34 5
|
16天前
|
设计模式 前端开发 Java
步步深入SpringMvc DispatcherServlet源码掌握springmvc全流程原理
通过对 `DispatcherServlet`源码的深入剖析,我们了解了SpringMVC请求处理的全流程。`DispatcherServlet`作为前端控制器,负责请求的接收和分发,处理器映射和适配负责将请求分派到具体的处理器方法,视图解析器负责生成和渲染视图。理解这些核心组件及其交互原理,有助于开发者更好地使用和扩展SpringMVC框架。
27 4
|
27天前
|
缓存 Java 数据库连接
Spring框架中的事件机制:深入理解与实践
Spring框架是一个广泛使用的Java企业级应用框架,提供了依赖注入、面向切面编程(AOP)、事务管理、Web应用程序开发等一系列功能。在Spring框架中,事件机制是一种重要的通信方式,它允许不同组件之间进行松耦合的通信,提高了应用程序的可维护性和可扩展性。本文将深入探讨Spring框架中的事件机制,包括不同类型的事件、底层原理、应用实践以及优缺点。
62 8
|
2月前
|
存储 Java 关系型数据库
在Spring Boot中整合Seata框架实现分布式事务
可以在 Spring Boot 中成功整合 Seata 框架,实现分布式事务的管理和处理。在实际应用中,还需要根据具体的业务需求和技术架构进行进一步的优化和调整。同时,要注意处理各种可能出现的问题,以保障分布式事务的顺利执行。
74 6
|
2月前
|
前端开发 Java 开发者
Spring MVC中的请求映射:@RequestMapping注解深度解析
在Spring MVC框架中,`@RequestMapping`注解是实现请求映射的关键,它将HTTP请求映射到相应的处理器方法上。本文将深入探讨`@RequestMapping`注解的工作原理、使用方法以及最佳实践,为开发者提供一份详尽的技术干货。
131 2