前言
- springMVC依赖。
<dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>5.2.0.RELEASE</version></dependency>
一、springMVC
1.MVC模型
- 模型(models),视图(view),控制器(controller),是一种软件设计的规范。
- 把业务逻辑,视图,数据分离开来组织代码。
- 降低视图和业务逻辑的耦合性。
2.springMVC的执行过程
- 用户发送请求,DispatcherServle控制器接收请求并拦截请求。
- DispatcherServlet调用HandlerMapping,查找具体的Handler。
- HandlerExecution是具体的Handler,主要作用是根据请求的url找到具体的控制器(controller),HandlerExecution将解析后的信息返回给DispatcherServlet。
- DispatcherServlet调用HandlerAdapter,HandlerAdapter按照特定的规则(实现controller接口)执行Handler。
- Handler会让具体的控制器(controller)执行。
- 控制器(controller)执行完成后返回给HandlerAdapter一个ModelAndView。
- HandlerAdapter把ModerAndView返回给DispatcherServlet。
- DispatherServlet把ModelAndView传给ViewResolver。
- ViewResolver执行完成后返回给DispatherServlet一个具体的View。
- viewResolver获取mv的数据;
- 解析mv的视图名字;
- 拼接视图名字找到对应的视图;
- 将数据渲染到视图上。
- DispatcherServlet根据视图解析器解析的视图结果,调用具体的视图。
- DispatcherServlet把视图响应呈现给用户。

3.springMVC的配置文件的方式
web.xml
<?xml version="1.0" encoding="UTF-8"?><web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"version="4.0"><!--配置DispatcherServlet: 这个是SpringMVC的核心:请求分发器,前端控制器--><servlet><servlet-name>springmvc</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:springmvc-servlet.xml</param-value></init-param><!--启动级别:1--><load-on-startup>1</load-on-startup></servlet><!--在SpringMVC中, / /*/ : 只匹配所有的请求,不会去匹配jsp页面/* : 匹配所有的请求,包括jsp页面--><servlet-mapping><servlet-name>springmvc</servlet-name><url-pattern>/</url-pattern></servlet-mapping></web-app>
- springmvc-servlet.xml
<?xml version="1.0" encoding="UTF-8" ?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd"><!--处理器映射器--><bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean><!--处理器适配器--><bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"></bean><!--视图解析器:模板引擎可以替换,Thymeleaf,Freemarker...--><bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver"><property name="prefix" value="/WEB-INF/jsp/"/><property name="suffix" value=".jsp"/></bean><!--BeanNameUrlHandlerMapping:bean--><bean id="/hello" class="com.sywl.servlet.HelloServlet"></bean></beans>
4.springMVC的注解配置的方式
springMVC的web.xml的配置不变
springMVC的springmvc-servlet.xml的配置如下:
<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xsi:schemaLocation="http://www.springframework.org/schema/beanshttps://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttps://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttps://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--自动扫描包,让指定包下的注解生效,由IOC容器统一管理--><context:component-scan base-package="com.shuai.controller"/><!--tomcat提供了默认的Servlet来处理静态资源--><mvc:default-servlet-handler/><!--帮助我们完成DefaultAnnotationHandlerMapping和AnnotationMethodHandlerAdapter实例的注册,这两个实例是为了让@RequestMapping注解生效--><mvc:annotation-driven/><!--视图控制器--><bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver"><!--前缀--><property name="prefix" value="/WEB-INF/jsp/"/><!--后缀--><property name="suffix" value=".jsp"/></bean></beans>
5.springMVC的转发和重定向
- 跳转页面通过ModelAndView
设置ModelAndView对象,根据view的名称和视图解析器跳转到指定的页面。
页面:{视图解析器前缀}+viewName+{视图解析器后缀}public class HelloController implements Controller {public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {ModelAndView mv = new ModelAndView();// 业务代码String result = "HelloSpringMVC";mv.addObject("msg",result);// 试图跳转mv.setViewName("test");return mv;}}
- 通过设置ServletAPI,不需要视图解析器
- 通过HttpServletResponse进行输出:resp.getWriter().println(“Hello,Spring By Servlet API”);
- 通过HttpServletResponse实现重定向:resp.sendRedirect(“/index.jsp”);
- 通过HttpServletResponse实现转发:resp.getRequestDispatcher(“/WEB-INF/jsp/test.jsp”).forward(req,resp);
- springMVC的转发和重定向
6.springMVC接收请求参数
- 前端传递的参数名和方法上的参数名相同,可以直接接收参数使用。
- 前端传递的参数名和方法上的参数名不相同,可以使用@RequestParam指定前端传递的参数名和方法上的参数名绑定。
@RequestMapping("/add")public String test(@RequestParam("username") String name){// @RequestParam限定前端传递的参数名只能是username并且和方法上的参数名name绑定。// 如果前端传递的参数名不是username,则会报错。return "test";}
前端传递对象时,方法上可以用实体类进行接收。
(前端传递的一个对象User,后端User实体类匹配传递来的User对象中的字段名:如果名字一致则接收参数,否则匹配不到)@GetMapping("/t2")public String test2(User user, Model model){// 1.接收前端参数System.out.println("前端接收到的参数为:"+user);model.addAttribute("msg",user);return "test";}
7.springMVC的数据回显
- Model:简化版,只有简单的几个方法适用于存储数据,对新手友好。
- ModelMap:继承了LinkedMap,除自身方法,还继承了LinkedMap的方法和特性。
- ModelAndView:可以在存储数据的同时,设置返回的逻辑视图,控制展示的跳转。
8.springMVC的乱码解决
post提交中文乱码的解决,一般配置springMVC过滤器就可以解决。实在解决不掉,把视频再看一遍。
<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><!--注意是/*--><!--/* : 匹配所有的请求,包括jsp页面--><url-pattern>/*</url-pattern></filter-mapping>
二、Restful风格
Restful就是一个资源定位及资源操作的风格。不是标准也不是协议,只是一种风格。
基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。
- 使用RESTful操作资源:可以通过不同的请求方式来实现不同的效果!比如:请求地址一样,但是功能可以不同。
- GetMapping
- PostMapping
- DeleteMapping
- PutMapping
- 路径参数
@PathVariable@RequestMapping("add/{a}/{b}")public String test1(@PathVariable int a,@PathVariable int b, Model model){int res = a + b;model.addAttribute("msg","结果为:"+res);return "test";}
三、JSON
1.json
- json是一种轻量级的数据交换格式。
- json和js对象的关系:json是js对象的字符串表示法,json本质上就是一个字符串。
- json和js对象的互相转换。
- 对象的键名也可以使用引号包裹。
var obj = JSON.parse('{"name":"zs","age":"12"}')// {name:"zs",age:12} 反序列化var json = JSON.Stringify({name:"zs",age:12})// '{"name":"zs","age":"12"}' 序列化
2.Jackson(对象转成json字符串)
- 导入Jackson包。
<dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.9.9.2</version></dependency>
- springmvc的配置文件配置中文乱码的问题。
<!--jackson乱码问题配置--><mvc:annotation-driven><mvc:message-converters><bean class="org.springframework.http.converter.StringHttpMessageConverter"><constructor-arg value="UTF-8"/></bean><bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"><property name="ObjectMapper"><bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean"><property name="failOnEmptyBeans" value="false"/></bean></property></bean></mvc:message-converters></mvc:annotation-driven>
写工具包JsonUtil
public class JsonUtil {public static String getJson(Object obj){return getJson(obj,"yy-MM-dd HH:mm:ss");}public static String getJson(Object obj,String dateFormat){// 创建一个解析器对象ObjectMapper mapper = new ObjectMapper();// 不使用时间戳的方式mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS,false);// 自定义日期格式对象SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);// 指定日期格式mapper.setDateFormat(sdf);try {return mapper.writeValueAsString(obj);} catch (JsonProcessingException e) {e.printStackTrace();}return null;}}
我们使用工具类,代码就更加简洁了
@RequestMapping("/j5")public String json5(){Date date = new Date();String json = JsonUtil.getJson(date);return json;}
2.Fastjson(对象转成json字符串)
- 导入Fastjson包。
<dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.60</version></dependency>
使用
@RequestMapping("/j6")public String json6() throws JsonProcessingException {// 创建一个对象User user1 = new User("小帅一号", 21, "男");User user2 = new User("小帅二号", 21, "男");User user3 = new User("小帅三号", 21, "男");User user4 = new User("小帅四号", 21, "男");ArrayList<User> list = new ArrayList<User>();list.add(user1);list.add(user2);list.add(user3);list.add(user4);String str = JSON.toJSONString(list);return str;}
Fastjson的三个主要的类
- JSONObject 代表json对象
- JSONObject实现了Map接口.猜想JSONObject底层操作是由Map实现的
- JSONObject对应json对象,通过各种形式的get()方法可以获取json对象中的数据,也可利用size(), isEmpty()等方法获取”键:值”对的个数和判断是否为空,其本质是通过实现Map接口并调用接口中的方法完成的.
- JSONArray 代表json对象数组
- 内部是有List接口中的方法来完成操作的。
- JSON代表 JSONObject和JSONArray的转化
- JSON类源码分析与使用
- 仔细观察这些方法,主要是实现json对象,json对象数组,javabean对象,json字符串之间的相互转化。
四、AJax
- AJAX是Asynchronous JavaScript and XML(异步的JavaScript 和 XML)。
- AJAX是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术。
- Ajax不是一种新的编程语言,而是一种用于创建更好更快以及交互性更强的Web应用程序的技术。
了解%2C#card=math&code=.ajax%28%7B%7D%29%2A%2A%2C%2A%2A&id=KsMiS).post({}),$.get({}) 请求的区别
https://blog.csdn.net/csucsgoat/article/details/117726102
$.ajax({url:"${pageContext.request.contextPath}/a1", // 请求urldata:{'name':$("#name").val()}, // 携带的数据datasuccess:function (data) { // 回调函数});
五、拦截器
1.概念
SpringMVC的处理器拦截器类似于Servlet开发中的过滤器Filter,用于对处理器进行预处理和后处理。开发者可以自己定义一些拦截器来实现特定的功能。
2.拦截器和过滤器的区别
2.1过滤器
- servlet规范中的一部分,在任何的Javaweb工程中都可以使用。
- 在中配置了/*,可以对所有要访问的资源进行拦截。
2.2拦截器
- 拦截器是springMVC框架自己的,只有使用了springMVC框架的工程才能使用。(最早在struct框架中就有)
- 拦截器只会拦截访问的控制器方法,如果访问的是html/css/js/jsp/image则不会进行拦截。
- 拦截器是AOP思想的具体应用。
3.拦截器配置和使用
在logininterceptor包下编写拦截器函数
public class LoginInterceptor implements HandlerInterceptor {public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException {// 如果是登陆页面则放行System.out.println("uri: " + request.getRequestURI());if (request.getRequestURI().contains("login")) {return true;}HttpSession session = request.getSession();// 如果用户已登陆也放行if(session.getAttribute("user") != null) {return true;}// 用户没有登陆跳转到登陆页面request.getRequestDispatcher("/WEB-INF/jsp/login.jsp").forward(request, response);return false;}public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {}public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {}}
- 在springmvc-servlet.xml中配置
<mvc:interceptors><mvc:interceptor><mvc:mapping path="/**"/><bean id="loginInterceptor" class="com.shuai.interceptor.LoginInterceptor"/></mvc:interceptor></mvc:interceptors>
六、文件上传和下载
1.前端表单要求
文件上传是项目开发中最常见的功能之一 ,springMVC 可以很好的支持文件上传,但是SpringMVC上下文中默认没有装配MultipartResolver,因此默认情况下其不能处理文件上传工作。如果想使用Spring的文件上传功能,则需要在上下文中配置MultipartResolver。
前端表单要求:为了能上传文件,必须将表单的method设置为POST,并将enctype设置为multipart/form-data。只有在这样的情况下,浏览器才会把用户选择的文件以二进制数据发送给服务器;
对表单中的 enctype 属性做个详细的说明:
- application/x-www=form-urlencoded:默认方式,只处理表单域中的 value 属性值,采用这种编码方式的表单会将表单域中的值处理成 URL 编码方式。
- multipart/form-data:这种编码方式会以二进制流的方式来处理表单数据,这种编码方式会把文件域指定文件的内容也封装到请求参数中,不会对字符编码。
- text/plain:除了把空格转换为 “+” 号外,其他字符都不做编码处理,这种方式适用直接通过表单发送邮件。
<form action="" enctype="multipart/form-data" method="post"><input type="file" name="file"/><input type="submit"></form>
一旦设置了enctype为multipart/form-data,浏览器即会采用二进制流的方式来处理表单数据,而对于文件上传的处理则涉及在服务器端解析原始的HTTP响应。在2003年,Apache Software Foundation发布了开源的Commons FileUpload组件,其很快成为Servlet/JSP程序员上传文件的最佳选择。
- Servlet3.0规范已经提供方法来处理文件上传,但这种上传需要在Servlet中完成。
- 而Spring MVC则提供了更简单的封装。
- Spring MVC为文件上传提供了直接的支持,这种支持是用即插即用的MultipartResolver实现的。
- Spring MVC使用Apache Commons FileUpload技术实现了一个MultipartResolver实现类:
- CommonsMultipartResolver。因此,SpringMVC的文件上传还需要依赖Apache Commons FileUpload的组件。
2.后端的要求
- 导入包
<!--文件上传--><dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.3.3</version></dependency><!--servlet-api导入高版本的--><dependency><groupId>javax.servlet</groupId><artifactId>javax.servlet-api</artifactId><version>4.0.1</version></dependency>
- springMVC文件上传配置
<!--文件上传配置--><bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"><!-- 请求的编码格式,必须和jSP的pageEncoding属性一致,以便正确读取表单的内容,默认为ISO-8859-1 --><property name="defaultEncoding" value="utf-8"/><!-- 上传文件大小上限,单位为字节(10485760=10M) --><property name="maxUploadSize" value="10485760"/><property name="maxInMemorySize" value="40960"/></bean>
- 编写前端页面
<form action="/upload" enctype="multipart/form-data" method="post"><input type="file" name="file"/><input type="submit" value="upload"></form>
Controller
@Controllerpublic class FileController {//@RequestParam("file") 将name=file控件得到的文件封装成CommonsMultipartFile 对象//批量上传CommonsMultipartFile则为数组即可@RequestMapping("/upload")public String fileUpload(@RequestParam("file") CommonsMultipartFile file , HttpServletRequest request) throws IOException {//获取文件名 : file.getOriginalFilename();String uploadFileName = file.getOriginalFilename();//如果文件名为空,直接回到首页!if ("".equals(uploadFileName)){return "redirect:/index.jsp";}System.out.println("上传文件名 : "+uploadFileName);//上传路径保存设置String path = request.getServletContext().getRealPath("/upload");//如果路径不存在,创建一个File realPath = new File(path);if (!realPath.exists()){realPath.mkdir();}System.out.println("上传文件保存地址:"+realPath);InputStream is = file.getInputStream(); //文件输入流OutputStream os = new FileOutputStream(new File(realPath,uploadFileName)); //文件输出流//读取写出int len=0;byte[] buffer = new byte[1024];while ((len=is.read(buffer))!=-1){os.write(buffer,0,len);os.flush();}os.close();is.close();return "redirect:/index.jsp";}}
- 测试上传文件,OK!
采用file.Transto 来保存上传的文件
- 编写Controller ```java /*
采用file.Transto 来保存上传的文件 */ @RequestMapping(“/upload2”) public String fileUpload2(@RequestParam(“file”) CommonsMultipartFile file, HttpServletRequest request) throws IOException {
//上传路径保存设置 String path = request.getServletContext().getRealPath(“/upload”); File realPath = new File(path); if (!realPath.exists()){
realPath.mkdir();
} //上传文件地址 System.out.println(“上传文件保存地址:”+realPath);
//通过CommonsMultipartFile的方法直接写文件(注意这个时候) file.transferTo(new File(realPath +”/“+ file.getOriginalFilename()));
return “redirect:/index.jsp”; } ```
- 前端表单提交地址修改
- 访问提交测试,OK!
文件下载
狂神说Java

