0、测试环境
我们简化开发,创建一个简单的环境(因为没有其它包比如 service、dao,所以这里不用 Spring 容器,只用 SpringMVC 容器):
Servelet 容器配置:
package com.lyh.config;import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;/*** Servlet 容器配置类*/
public class ServletContainersInitConfig extends AbstractAnnotationConfigDispatcherServletInitializer {protected Class<?>[] getRootConfigClasses() {return new Class[0];}protected Class<?>[] getServletConfigClasses() {return new Class[]{SpringMvcConfig.class};}protected String[] getServletMappings() {return new String[]{"/"};}
}
注意:有了这个就不需要 web.xml 了,不然启动 tomcat 会报错!!!
SpringMVC 配置:
package com.lyh.config;import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;@Configuration
@ComponentScan("com.lyh.controller") // 扫描
public class SpringMvcConfig {}
UserController:
package com.lyh.controller;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;@Controller
public class UserController {@RequestMapping("/save")@ResponseBodypublic String save(){return "{'module':'user save'}";}@RequestMapping("/delete")@ResponseBodypublic String delete(){return "{'module':'user delete'}";}
}
BookController:
package com.lyh.controller;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;@Controller
public class BookController {@RequestMapping("/save")@ResponseBodypublic String save(){return "{'module':'book save'}";}}
1、请求映射路径
在上面的环境中,我们可看到,对于不同 controller 类中存在有相同映射路径(/save),这是我们开发中经常会遇到的问题,也就是不同开发人员设置了相同的映射路径该怎么办?
解决办法:设置模块名作为请求路径前缀。
这样,我们就解决了不同业务模块请求路径冲突的问题了,但是我们发现,如果这个每个业务模块下面的方法前面都加一个( /module )是不是太冗余了。所以,SpringMVC 还能帮我们继续简化开发( 直接在 Controller 类上面定义前缀):
这里我们就可以总结一下注解 @RequestMapping 的用法了:
名称 | 类型 | 位置 | 作用 |
@RequestMapping | 类注解 / 方法注解 | 控制器类 / 控制器方法上面 | 设置当前控制器方法请求访问路径,如果加在控制器类上则代表统一访问路径前缀 |
2、请求方式
2.1、Get 请求
Get 请求非常简单,我们只需要在访问路径后面跟上一个 ?参数名=参数值 即可,如果是多个参数,直接 ?参数名1=参数值1&参数名2=参数值2 即可。
注意:我们之前在学习 Servlet 的时候,Get 和 Post 请求是区分的,但是在 SpringMVC 中,我们并不需要在方法中分开处理不同类型的请求。
2.2、Post 请求
这里用 Postman 来进行模拟发送 Post 请求:
注意:Post 请求发表单的时候,我们请求体中用的应该是 x-www-from-urlencoded 。
from-date 区别于 x-www-from-urlencoded 不同的是,它不仅可以发表单,还可以发送文件:
2.3、Post 请求中文乱码问题
中文乱码的问题我们之前在学习 Servlet 的时候是通过过滤器来实现的,在 SpringMVC 这里是一样的。我们不需要在负责响应的方法里去写,而是直接去 Servlet 容器配置类里重写一下 getServletFilters 方法即可:
注意:这里的过滤器只对 Post 请求有效,Get 请求设置过滤器要复杂一些。
2.4、请求参数
参数种类:
- 普通参数
- POJO
- 嵌套POJO
- 数组
- 集合
2.4.1、客户端和服务端参数名不一致问题
也就是客户端请求参数和我们服务端 Controller 类下面的方法形参名不一致,这种问题很常见,解决办法也很简单,直接通过 @RequestParam("参数名") 把客户端的参数名和我们服务端方法的形参绑定在一起:
但是需要注意的是:设置了这个注解就相当于指定了客户端的参数名,也就是说客户端要想再用 name 来给服务端传递参数就不行了!
2.4.2、POJO 类型参数
我们先创建一个包 domain 并创建一个简单的 User 类:
package com.lyh.domain;public class User {private String name;private int age;@Overridepublic String toString() {return "User{" +"name='" + name + '\'' +", age=" + age +'}';}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}
在 Controller 中添加一个用 Pojo 作为参数传递的方法:
@RequestMapping("/saveByPojo")
@ResponseBody
public String saveByPojo(User user){System.out.println("普通参数传递 name = " + user.getName() + " , age = " + user.getAge());return "{'module':'user save'}";
}
注意:在Spring MVC中,不允许一个Controller下的不同方法具有相同的映射(我原本想着方法重载的,但是发现不行)。
运行结果和之前的普通参数传递是一致的,而我们并没有做额外的操作,只是简单的修改了参数类型,SpringMVC 会帮我们自动通过对象的参数名去找到我们的方法形参。所以,这就要求我们 POJO 的参数名必须和方法形参名一致!
2.4.3、嵌套 POJO
嵌套 POJO 也就是一个对象的参数是另一个对象。
我们再创建一个 Address 类(两个属性:province 和 city,这里省略代码)并添加给 User 类。
2.4.4、数组类型
同样非常简单,在控制器类下面添加一个方法:
发送 Post 请求:
这样,我们就通过多个相同参数名的方式实现了数组的传递。
2.4.5、集合类型
我们试着用和数组一样的方式去给集合传递参数:
在控制器类下创建一个方法:
Post 请求测试:
运行结果:
可以看到,服务器报了一个构造器的错误,说明服务器在尝试调用 List 的构造方法,但是我们知道 List 是一个接口,根本没有构造器。这也说明,我们上一节在用 Pojo 作为参数传递的时候,SpringMVC 底层其实是会调用形参的那个 Pojo 对象的构造器,然后通过 set 方法进行属性注入。
解决的办法也很简单,直接通过@RequestParam 注解把我们的请求参数当做集合的元素传递给集合(如果形参前面有 @RequestParam 注解,SpringMVC 会自动识别,而不会把它当做 Pojo 通过构造器创建空对象,然后通过 set 注入属性):
运行结果:
这样就没问题了。
2.5、JSON 传参
我们实际开发用到 JSON 的情况会很多很多,所以这里需要重点学习 JSON 的传参方法。
明天更新