在线购物网站建设流程,网络公司怎么优化网站,自动添加内部链接的wordpress插件,西安网站设计西安搜推宝一、什么是MVC
1.1概述 MVC是模型(Model)、视图(View)、控制器(Controller)的简写#xff0c;是一种软件设计规范。 是将业务逻辑、数据、显示分离的方法来组织代码。 MVC主要作用是降低了视图与业务逻辑间的双向偶合。 MVC不是一种设计模式#xff0c;MVC是一种架构模式…一、什么是MVC
1.1概述 MVC是模型(Model)、视图(View)、控制器(Controller)的简写是一种软件设计规范。 是将业务逻辑、数据、显示分离的方法来组织代码。 MVC主要作用是降低了视图与业务逻辑间的双向偶合。 MVC不是一种设计模式MVC是一种架构模式。当然不同的MVC存在差异。 Model模型数据模型提供要展示的数据因此包含数据和行为可以认为是领域模型或JavaBean组件包含数据和行为不过现在一般都分离开来Value Object数据Dao 和 服务层行为Service。也就是模型提供了模型数据查询和模型数据的状态更新等功能包括数据和业务。 View视图负责进行模型的展示一般就是我们见到的用户界面客户想看到的东西。 Controller控制器接收用户请求委托给模型进行处理状态改变处理完毕后把返回的模型数据返回给视图由视图负责展示。也就是说控制器做了个调度员的工作。 最典型的MVC就是JSP servlet javabean的模式。 职责分析 Controller控制器 取得表单数据 调用业务逻辑 转向指定的页面 Model模型 业务逻辑 保存数据的状态 View视图 显示页面 1.2Servlet
环境配置 dependenciesdependencygroupIdjunit/groupIdartifactIdjunit/artifactIdversion4.12/version/dependencydependencygroupIdorg.springframework/groupIdartifactIdspring-webmvc/artifactIdversion5.1.9.RELEASE/version/dependencydependencygroupIdjavax.servlet/groupIdartifactIdservlet-api/artifactIdversion2.5/version/dependency!-- https://mvnrepository.com/artifact/jakarta.servlet/jakarta.servlet-api --dependencygroupIdjakarta.servlet/groupIdartifactIdjakarta.servlet-api/artifactIdversion6.0.0/versionscopeprovided/scope/dependency!-- https://mvnrepository.com/artifact/jakarta.servlet.jsp.jstl/jakarta.servlet.jsp.jstl-api --dependencygroupIdjakarta.servlet.jsp.jstl/groupIdartifactIdjakarta.servlet.jsp.jstl-api/artifactIdversion3.0.0/version/dependency!-- https://mvnrepository.com/artifact/org.glassfish.web/jakarta.servlet.jsp.jstl --dependencygroupIdorg.glassfish.web/groupIdartifactIdjakarta.servlet.jsp.jstl/artifactIdversion3.0.1/version/dependencydependencygroupIdjavax.servlet/groupIdartifactIdjstl/artifactIdversion1.2/version/dependency/dependencies
编写一个Servlet类
package com.yanyu.test;import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;import java.io.IOException;WebServlet(/hello)
//实现Servlet接口
public class HelloServlet extends HttpServlet {Overrideprotected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {//取得参数String method req.getParameter(method);if (method.equals(add)){req.getSession().setAttribute(msg,执行了add方法);}if (method.equals(delete)){req.getSession().setAttribute(msg,执行了delete方法);}//业务逻辑//视图跳转req.getRequestDispatcher(/WEB-INF/jsp/hello.jsp).forward(req,resp);}Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {doGet(req,resp);}
}编写Hello.jsp
% page contentTypetext/html;charsetUTF-8 languagejava %
html
headmeta charsetUTF-8titleKuangshen/title
/head
body
${msg}
/body
/html
二、什么是SpringMVC 2.1Spring MVC的特点 轻量级简单易学 高效 , 基于请求响应的MVC框架 与Spring兼容性好无缝结合 约定优于配置 功能强大RESTful、数据验证、格式化、本地化、主题等 简洁灵活 Spring的web框架围绕DispatcherServlet [ 调度Servlet ] 设计。 DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始使用Java 5或者以上版本的用户可以采用基于注解形式进行开发十分简洁 正因为SpringMVC好 , 简单 , 便捷 , 易学 , 天生和Spring无缝集成(使用SpringIoC和Aop) , 使用约定优于配置 . 能够进行简单的junit测试 . 支持Restful风格 .异常处理 , 本地化 , 国际化 , 数据验证 , 类型转换 , 拦截器 等等......所以我们要学习 . 2.2中心控制器
Spring的web框架围绕DispatcherServlet设计。DispatcherServlet的作用是将请求分发到不同的处理器。从Spring 2.5开始使用Java 5或者以上版本的用户可以采用基于注解的controller声明方式。
Spring MVC框架像许多其他MVC框架一样, 以请求为驱动 , 围绕一个中心Servlet分派请求及提供其他功能DispatcherServlet是一个实际的Servlet (它继承自HttpServlet 基类)。 SpringMVC的原理如下图所示
当发起请求时被前置的控制器拦截到请求根据请求参数生成代理请求找到请求对应的实际控制器控制器处理请求创建数据模型访问数据库将模型响应给中心控制器控制器使用模型与视图渲染视图结果将结果返回给中心控制器再将结果返回给请求者。
2.3SpringMVC执行原理 图为SpringMVC的一个较完整的流程图实线表示SpringMVC框架提供的技术不需要开发者实现虚线表示需要开发者实现。
简要分析执行流程 DispatcherServlet表示前置控制器是整个SpringMVC的控制中心。用户发出请求DispatcherServlet接收请求并拦截请求。 我们假设请求的url为 : http://localhost:8080/SpringMVC/hello 如上url拆分成三部分 http://localhost:8080服务器域名 SpringMVC部署在服务器上的web站点 hello表示控制器 通过分析如上url表示为请求位于服务器localhost:8080上的SpringMVC站点的hello控制器。 HandlerMapping为处理器映射。DispatcherServlet调用HandlerMapping,HandlerMapping根据请求url查找Handler。 HandlerExecution表示具体的Handler,其主要作用是根据url查找控制器如上url被查找控制器为hello。 HandlerExecution将解析后的信息传递给DispatcherServlet,如解析控制器映射等。 HandlerAdapter表示处理器适配器其按照特定的规则去执行Handler。 Handler让具体的Controller执行。 Controller将具体的执行信息返回给HandlerAdapter,如ModelAndView。 HandlerAdapter将视图逻辑名或模型传递给DispatcherServlet。 DispatcherServlet调用视图解析器(ViewResolver)来解析HandlerAdapter传递的逻辑视图名。 视图解析器将解析的逻辑视图名传给DispatcherServlet。 DispatcherServlet根据视图解析器解析的视图结果调用具体的视图。 最终视图呈现给用户。
2.4Java Json解析框架 这里有四个常用的Java JSON解析框架GsonFastJsonJackson和Json-lib。这些框架都有各自的优缺点。其中Gson是功能最全的JSON解析器之一无需额外的jar包就能直接运行在JDK上。FastJson是阿里巴巴公司开发的高性能JSON处理器采用独创的算法将parse的速度提升到极致超过所有JSON库。Jackson是当前用得比较广泛的Java开源框架它所依赖的jar包较少简单易用。Json-lib最开始也是应用最广泛的JSON解析工具之一但现在在功能和性能上都不能满足互联网化的需求。 2.5相关注解 Controller注解 SpringMVC中Controller注解是用来标记一个类是一个控制器也就是一个处理请求的组件。Controller注解可以让一个普通的Java类变成一个SpringMVC的控制器而不需要实现任何接口或继承任何类。Controller注解可以和RequestMapping注解配合使用来定义请求和控制器方法之间的映射关系。Controller注解还可以和其他注解一起使用来实现更多的功能比如RequestParam、PathVariable、ModelAttribute等。 RestController注解 SpringMVC中RestController注解是用来标记一个类是一个RESTful风格的控制器也就是说该类中的所有方法的返回值都会直接写入HTTP响应体中而不需要经过视图解析器。RestController注解相当于Controller和ResponseBody注解的组合可以简化开发过程。RestController注解可以和RequestMapping注解配合使用来定义请求和控制器方法之间的映射关系。RestController注解还可以和其他注解一起使用来实现更多的功能比如PathVariable、RequestParam、RequestBody等。 RequestMapping注解 SpringMVC中RequestMapping注解是用来将请求和处理请求的控制器方法关联起来建立映射关系。RequestMapping注解可以标注在类或者方法上用来定义请求的URL地址、请求的方式、请求的参数、请求的头部等。RequestMapping注解还可以和其他注解一起使用来实现更多的功能比如PathVariable、RequestParam、RequestBody等。 三、第一个程序
实体类
package com.example.springmvcexamples.example01.entity;import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data;
import lombok.NoArgsConstructor;import java.time.LocalDateTime;Data
NoArgsConstructor
public class Address {private Integer id; // 地址的IDprivate String detail; // 地址的详细信息private String comment; // 地址的备注private User user; // 地址所属的用户JsonFormat(pattern yyyy-MM-dd HH:mm:ss) // 指定日期时间的格式化模式private LocalDateTime inertTime; // 地址的创建时间public Address(Integer id, String detail, String comment, LocalDateTime inertTime) {this.id id;this.detail detail;this.comment comment;this.inertTime inertTime;}
}package com.example.springmvcexamples.example01.entity;import lombok.Data;
import lombok.NoArgsConstructor;Data
NoArgsConstructor
public class User {private Integer id;
}控制类
package com.example.springmvcexamples.example01;import com.example.springmvcexamples.example01.entity.Address;
import com.example.springmvcexamples.vo.ResultVO;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpHeaders;
import org.springframework.web.bind.annotation.*;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.time.LocalDateTime;
import java.util.List;
import java.util.Map;Slf4j
RestController
RequestMapping(/api/example01/)
public class ExampleController01 {// 返回一个 ResultVO 对象其中包含一个名为 name 的键和值为 SUN 的映射GetMapping(index)public ResultVO getIndex() {return ResultVO.success(Map.of(name, SUN));}// 返回一个 ResultVO 对象其中包含一个名为 addresses 的键和值为 ADDRESSES 的映射GetMapping(addresses)public ResultVO getAddresses() {return ResultVO.success(Map.of(addresses, ADDRESSES));}// 接收一个 Address 对象作为请求体并打印 address 的 detail 和 comment 属性PostMapping(addresses)public ResultVO postAddress(RequestBody Address address) {log.debug(address.getDetail());log.debug(address.getComment());return ResultVO.success(Map.of());}// 接收一个 Address 对象作为请求体并打印 address 的 detail、comment 和 user 的 id 属性PostMapping(addresses02)public ResultVO postAddress2(RequestBody Address address) {log.debug(address.getDetail());log.debug(address.getComment());log.debug({}, address.getUser().getId());return ResultVO.success(Map.of());}// 根据 aid 的值从 ADDRESSES 中查找对应的 Address 对象并返回一个 ResultVO 对象其中包含一个名为 address 的键和对应的 Address 对象的映射GetMapping(addresses/{aid})public ResultVO getAddress(PathVariable(aid) int aid) {Address address ADDRESSES.stream().filter(a - a.getId() aid).findFirst().orElse(new Address());return ResultVO.success(Map.of(address, address));}// 打印请求的 URI 和请求头的信息GetMapping(inject)public void inject(HttpServletRequest request,HttpServletResponse response,RequestHeader HttpHeaders headers) {log.debug(request.getRequestURI());log.debug(String.valueOf(headers));}Autowiredprivate ObjectMapper mapper;// 使用 RequestParam 接收传统的 ? 传参打印接收到的 address 参数并将其转换为 Address 对象并打印 detail 属性GetMapping(search)public ResultVO getJson(RequestParam String address) throws JsonProcessingException {log.debug(address);Address a mapper.readValue(address, Address.class);log.debug(a.getDetail());return ResultVO.success(Map.of(name, BO));}// 创建一个包含三个 Address 对象的列表private final ListAddress ADDRESSES create();private ListAddress create() {Address a1 new Address(1, 956, a, LocalDateTime.now());Address a2 new Address(2, 925, b, LocalDateTime.now());Address a3 new Address(3, 121, null, null);return List.of(a1, a2, a3);}
}测试
###
GET http://localhost:8080/api/example01/index###
GET http://localhost:8080/api/example01/addresses###
POST http://localhost:8080/api/example01/addresses
Content-Type: application/json{detail: 956,comment: 测试
}###
GET http://localhost:8080/api/example01/addresses/3###
POST http://localhost:8080/api/example01/addresses02
Content-Type: application/json{detail: 956,comment: 测试,user3: {id: 10}
}
###
GET http://localhost:8080/api/example01/inject### 传统?传参支持传递json字符串
GET http://localhost:8080/api/example01/search?address{detail: 12}