SpringMVC(1)
一、三层架构
表现层:Web层,用于和客户端进行数据交互,采用MVC设计模型,Controller(SpringMVC)
业务层:处理业务逻辑,Service(Spring)
持久层:操作数据库,Dao(MyBatis)
二、入门案例
步骤:
- 在
pom.xml
中引入jar包 - 在
webapp/WEB-INF/web.xml
中配置核心控制器DispatcherServlet
- 在
src/main/resources
下创建SpringMVC.xml
配置文件 - 在
webapp下
创建首页页面index.jsp
- 在
src/main/java/controller
下创建HelloController控制器类 - 在
webapp/WEB-INF/pages
下创建跳转成功页面success.jsp
具体实现:
-
web.xml
:配置前端控制器和解决中文乱码的过滤器使用前端控制器servlet来加载后续的SprigMVC的xml配置文件 ==> 开启注解扫描 ==> Controller类启动 ==> 页面发送请求,Controller类中的方法生效 ==> 跳转成功页面
<web-app>
<display-name>Archetype Created Web Application</display-name>
<!-- 配置解决中文乱码的过滤器 -->
<filter>
<filter-name>encodingFilter</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>encodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- 前端控制器 -->
<servlet>
<servlet-name>dispatcherServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>dispatcherServlet</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
注意:在前端控制器里要配置初始化参数contextConfigLocation
,使得servlet启动时能够加载SpringMVC的xml配置文件
-
SpringMVC.xml
- 开启注解扫描 - 配置视图解析器 - 开启SpringMVC框架注解的支持
<!-- 开启注解扫描 -->
<context:component-scan base-package="com.Lemon"/>
<!-- 视图解析器
prefix:文件所在目录
suffix:文件的后缀
-->
<bean id="internalResourceViewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/pages/"/>
<property name="suffix" value=".jsp"/>
</bean>
<!-- 开启SpringMVC框架注解的支持 -->
<mvc:annotation-driven/>
index.jsp
:首页页面
<body>
<h3>入门程序</h3>
<!-- 超链接 -->
<a href="hello">入门程序</a>
</body>
HelloController.java
:控制器类,通过配置xml文件开启注解扫描后,只需要在该类上加上@Controller
注解,交给IOC容器管理即可
@Controller
public class HelloController {
/**
* 入门案例:
* jsp页面发送请求到后台,调用sayHello方法,需要加上@RequestMapping请求映射注解,并指定路径
*/
@RequestMapping(path = "/hello")
public String sayHello(){
System.out.println("Hello SpringMVC");
return "success"; // 如果方法生效了,就应该要跳转到success.jsp页面
}
-
sunccess.jsp
:成功页面当在首页点击超链接后,jsp页面发送请求到后端,
由于超链接的href
指向了sayHello()
方法的@RequestMapping
所指的路径,
所以,会调用sayHello()方法,方法生效执行成功return "success"
,
那么就会跳转到success.jsp
页面,
最终页面会显示success.jsp
的“入门成功”,并且后台打印sayHello()
方法的"Hello SpringMVC"。
<body>
<h3>入门成功</h3>
</body>
三、请求参数的绑定
- 提交表单的数据格式:
K=V
,中间用&
连接例如:username=hehe&password=123
- SpringMVC参数绑定过程:把表单提交的请求参数,作为Controller控制器类中的方法的参数进行绑定,要求表单的name必须和参数的名称相同
- 支持的数据类型
– 基本数据类型和字符串类型
具体代码:
首页param.jsp
:
<body>
<%-- (1)基本数据类型和字符串类型 --%>
<a href="param/testParam?username=hehe&password=123">请求参数的绑定</a>
</body>
控制器类ParamController.java
:
@Controller
@RequestMapping("/param")
public class ParamController {
// (1)基本数据类型和字符串类型
@RequestMapping("/testParam")
public String testParam1(String username, String password){
System.out.println("执行了...");
System.out.println("用户名:" + username);
System.out.println("密码:" + password);
return "success";
}
}
要求:超链接的name(href
的?username=hehe
)必须和方法参数String username
相同
效果:SpringMVC会将页面请求参数赋值给控制器类方法的形参,在后台会打印页面输入的username和password的具体数据hehe和123,页面会跳转到success页面
– 实体类类型JavaBean对象
具体代码:
param.jsp
:
<body>
<%-- (2)实体类型JavaBean --%>
<form action="param/saveAccount" method="post">
姓名:<input type="text" name="username"/><br/>
密码:<input type="text" name="password"/><br/>
金额:<input type="text" name="money"/><br/>
用户姓名:<input type="text" name="user.uname"/><br/>
用户年龄:<input type="text" name="user.age"/><br/>
<input type="submit" name="提交"/>
</form>
</body>
ParamController.java
:
@Controller
@RequestMapping("/param")
public class ParamController {
// (2)实体类型JavaBean
@RequestMapping("/saveAccount")
public String testParam2(Account account){
System.out.println("执行了...");
System.out.println(account);
return "success";
}
}
实体类Account.java
:加上get,set,toString方法
public class Account implements Serializable {
private String username;
private String password;
private Double money;
// (2)实体类型JavaBean
private User user;
}
实体类User.java
:加上get,set,toString方法
public class User implements Serializable {
private String uname;
private Integer age;
// 自定义类型转换器
private Date date;
}
要求:
- 提交表单中的name(
name="username"
)必须和实体类Account
的属性名相同。 - 如果实体类中有引用类型属性(另一个JavaBean对象
User user
),那么可以在表单中使用对象.属性user.uname
调用引用类型属性的属性 - 当页面提交表单数据,会调用属性的set方法设置数据;方法生效时,传入JavaBean对象,就可以调用get方法获得数据
– 集合数据类型(List,Map等)
具体代码:
param.jsp
:
<body>
<%-- (3)集合数据类型 --%>
<form action="param/saveAccount" method="post">
姓名:<input type="text" name="username" /><br/>
密码:<input type="text" name="password" /><br/>
金额:<input type="text" name="money" /><br/>
用户姓名:<input type="text" name="list[0].uname" /><br/>
用户年龄:<input type="text" name="list[0].age" /><br/>
用户姓名:<input type="text" name="map['one'].uname" /><br/>
用户年龄:<input type="text" name="map['one'].age" /><br/>
<input type="submit" value="提交" />
</form>
</body>
ParamController.java
:
@Controller
@RequestMapping("/param")
public class ParamController {
// (2)实体类型JavaBean
@RequestMapping("/saveAccount")
public String testParam2(Account account){
System.out.println("执行了...");
System.out.println(account);
return "success";
}
}
实体类Account.java
:加上get,set,toString方法
public class Account implements Serializable {
private String username;
private String password;
private Double money;
// (2)实体类型JavaBean
private User user;
// (3)集合数据类型(List,Map等)
private List<User> list;
private Map<String, User> map;
}
实体类User.java
:加上get,set,toString方法
public class User implements Serializable {
private String uname;
private Integer age;
// 自定义类型转换器
private Date date;
}
要求:
- 对于List类型,表单使用
list[0].uname
- 对于Map类型,表单使用
map['one'].uname
四、自定义类型转换器
表单提交的都是字符串类型,就算后台定义Integer类型,数据也可以封装上
这说明,Spring框架会默认进行数据类型转换
但是,如果是自定义的类型,那么需要自定义类型转换器
步骤:
* 1.定义一个类型转换的方法,并实现Convert接口
* implements Converter<S, T>:S是String类型,T是自定义类型
* 2.在SpringMVC.xml配置文件中配置自定义类型转换器
* 同时,要把自定义的类型转换的方法注册进来,需要在SpringMVC框架注解中注册自定义类型转换器
* ConversionServiceFactoryBean
* <bean class="com.Lemon.utils.StringToDateConverter"/>
* <mvc:annotation-driven conversion-service="conversionService"/>
具体代码:
自定义一个日期类型的转换器
因为SpringMVC默认只能将yyyy/MM/dd的字符串转换成日期
现在要求能将yyyy-MM-dd的字符串转换成日期。
StringToDateConverter.java
:把字符串转换成日期的自定义类型转换器
public class StringToDateConverter implements Converter<String, Date> {
@Override
public Date convert(String source) {
// 判断
if (source == null){
throw new RuntimeException("请您传入数据");
}
DateFormat df = new SimpleDateFormat("yyyy-MM-dd");
// 把字符串转换成日期
try {
return df.parse(source);
} catch (Exception e) {
throw new RuntimeException("数据类型转换出现错误");
}
}
}
springmvc.xml
:
<!-- 配置自定义类型转换器
1. 需要把自定义的类型转换的方法注册进来
2. 需要在SpringMVC框架注解中注册自定义类型转换器
-->
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<set>
<bean class="com.Lemon.utils.StringToDateConverter"/>
</set>
</property>
</bean>
<!-- 开启SpringMVC框架注解的支持 -->
<mvc:annotation-driven conversion-service="conversionService"/>
ParamController.java
:
@Controller
@RequestMapping("/param")
public class ParamController {
// 自定义类型转换器
@RequestMapping("/saveUser")
public String testParam3(User user){
System.out.println("执行了...");
System.out.println(user);
return "success";
}
}
五、获取原生的Servlet的API对象
只需要在Controller控制器类的方法的形参列表中定义HttpServletRequest
和HttpServletResponse
对象即可使用
ParamController.java
:
@Controller
@RequestMapping("/param")
public class ParamController {
// 获取原生的Servlet的API对象
@RequestMapping("/testServlet")
public String testServlet(HttpServletRequest request, HttpServletResponse response){
System.out.println("执行了...");
System.out.println(request);
HttpSession session = request.getSession();
System.out.println(session);
ServletContext servletContext = session.getServletContext();
System.out.println(servletContext);
System.out.println(response);
return "success";
}
}