目录
一、前后端分离开发与混合开发
1.1 混合开发模式
1.2 前后端分离模式【重点】
二、页面跳转控制
2.1 通过JSP实现页面跳转
2.2 转发与重定向
三、返回JSON数据
3.1 导包与配置
3.2 使用@ResponseBody
四、返回静态资源
4.1 为什么无法直接查询静态资源
4.2 配置实现静态资源查找
一个项目的开发可以分为两种开发模式,分别是前后端分离和混合开发。以下介绍两种开发模式的特点。
指将前端和后端的代码集成在同一个项目中,共享相同的技术栈和框架。这种模式在小型项目中比较常见,可以减少学习成本和部署难度。但是,在大型项目中,这种模式会导致代码耦合性很高,维护和升级难度较大。
对于混合开发,我们就需要使用动态页面技术,动态展示Java的共享域数据!!
指将前端的界面和后端的业务逻辑通过接口分离开发的一种方式。开发人员使用不同的技术栈和框架,前端开发人员主要负责页面的呈现和用户交互,后端开发人员主要负责业务逻辑和数据存储。前后端通信通过 API 接口完成,数据格式一般使用 JSON 或 XML。前后端分离模式可以提高开发效率,同时也有助于代码重用和维护。
这里页面跳转都是基于jsp技术。即混合开发模式下的应用。
1.使用jsp技术,需要导入以下依赖
jakarta.servlet.jsp.jstl
jakarta.servlet.jsp.jstl-api
3.0.0
2.jsp页面创建
建议位置:/WEB-INF/下,避免外部直接访问!
位置:/WEB-INF/views/home.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
${data}
1.为什么/WEB-INF/能够实现避免外部访问
这种限制的实现是由 Java Web 容器(比如 Tomcat、Jetty 等)来完成的,它们会拦截对 /WEB-INF/ 目录及其子目录的请求,阻止这些内容被直接访问。这样做的目的是为了保护应用程序的安全性和完整性,避免敏感信息和代码被非授权用户获取。
3. 快速响应模版页面
配置对应的mvc类
@EnableWebMvc //json数据处理,必须使用此注解,因为他会加入json处理器
@Configuration
@ComponentScan(basePackages = "com.atguigu.controller") //TODO: 进行controller扫描
//WebMvcConfigurer springMvc进行组件配置的规范,配置组件,提供各种方法! 前期可以实现
public class SpringMvcConfig implements WebMvcConfigurer {
//配置jsp对应的视图解析器
@Override
public void configureViewResolvers(ViewResolverRegistry registry) {
//快速配置jsp模板语言对应的前缀和后缀
registry.jsp("/WEB-INF/views/",".jsp");
}
}
handler返回视图
/**
* 跳转到提交文件页面 /save/jump
*
* 如果要返回jsp页面!
* 1.方法返回值改成字符串类型
* 2.返回逻辑视图名即可
*
* + 逻辑视图名 +
*
*/
@GetMapping("jump")
public String jumpJsp(HttpServletRequest request){
//设置共享域
request.setAttribute( s: "data", o: "hello jsp!!!");
model.addAttribute("data","]spController.index");
return "index";
}
运行后结果:
1. 由于在配置中加上了前缀和后缀,所以return的值只需要对应jsp页面的名称即可。
2. 混合开发模式下,前后端的数据交流是通过共享域进行的。
在springMvc中实现转发和重定向是通过关键词Forward和Redirect来实现的。
@RequestMapping("/redirect-demo")
public String redirectDemo() {
// 重定向到 /demo 路径
return "redirect:/demo";
}
@RequestMapping("/forward-demo")
public String forwardDemo() {
// 转发到 /demo 路径
return "forward:/demo";
}
//注意: 转发和重定向到项目下资源路径都是相同,都不需要添加项目根路径!填写项目下路径即可!
注意: 转发是只能访问项目下的资源,而重定向是可以重定向到其它资源,比如其它不属于本项目的网站。
可能会问,转发下既然访问项目下的资源,而tomcat在部署的时候设置了根路径,是否需要把根路径也一并导入进去?答案是不需要
springmvc对转发做了优化,即自动帮你加上了对应的根路径,而如果再转发时候写上根路径,就会导致重复的根路径的填写。
由于使用JSON返回数据使用到@ResponseBody,就不再使用页面跳转中的跳转(forward)和重定向(redirect)
使用JSON数据需要导入以下包
com.fasterxml.jackson.core
jackson-databind
2.15.0
写配置类
//TODO: SpringMVC对应组件的配置类 [声明SpringMVC需要的组件信息]
//TODO: 导入handlerMapping和handlerAdapter的三种方式
//1.自动导入handlerMapping和handlerAdapter [推荐]
//2.可以不添加,springmvc会检查是否配置handlerMapping和handlerAdapter,没有配置默认加载
//3.使用@Bean方式配置handlerMapper和handlerAdapter
@EnableWebMvc //json数据处理,必须使用此注解,因为他会加入json处理器
@Configuration
@ComponentScan(basePackages = "com.atguigu.controller") //TODO: 进行controller扫描
//WebMvcConfigurer springMvc进行组件配置的规范,配置组件,提供各种方法! 前期可以实现
public class SpringMvcConfig implements WebMvcConfigurer {
}
如果使用JSON格式数据与前端进行交互,那么也就意味着抛弃了混合开发的模式,所以需要用到@ResponseBody注解,这个注解让解析器不会去找视图层,而是直接返回数据。
对于handler方法而言,使用注解只需要在方法上进行定义。
@GetMapping("/accounts/{id}")
@ResponseBody
public Object handle() {
// ...
return obj;
}
@RequestMapping(value = "/user/detail", method = RequestMethod.POST)
@ResponseBody
public User getUser(@RequestBody User userParam) {
System.out.println("userParam = " + userParam);
User user = new User();
user.setAge(18);
user.setName("John");
//TODO:返回的对象,会使用jackson的序列化工具,转成json返回给前端!
return user;
}
通过POSTMAN工具访问后结果如下:
如果每一个方法都需要进行返回JSON格式数据,那么是否可以简化一下呢?打开注解源码
可以看到该注解是可以作用在类上的,也就意味,只需要在 类上直接加上注解即可
//TODO:只需要在类上加上@ResponseBody即可表示所有handler方法都是直接返回数据
@Controller
@ResponseBody
@RequestMapping("appointment")
public class AppointmentController {
@RequestMapping("Hello/{name}/{sex}")
public String hello(@PathVariable String name, @PathVariable String sex) {
System.out.println("name = " + name + ", sex = " + sex);
return "name = " + name + ", sex = " + sex;
}
@RequestMapping("/JsonTest")
// TODO:通过设置响应体注解@RequestBody实现接收json数据
public String Json(@RequestBody JSONObject jsonObject){
System.out.println("成功接受到JSON数据");
return "返回接收到的JSON数据为:"+jsonObject.toJSONString();
}
@RequestMapping("/save")
public String CookieTest(HttpServletResponse servletResponse){
Cookie cookie = new Cookie("CookieTestName","AlphaMilk");
servletResponse.addCookie(cookie);
System.out.println("成功设置好cookie");
return "设置cookie成功";
}
@RequestMapping("/GetCookie")
public String GetCookie(@CookieValue("CookieTestName") String value){
return "获取到的Cookie值为"+value;
}
@RequestMapping("/Header")
public String GetHead(@RequestHeader("HOST") String value){
return "获取到的HOST值为"+value;
}
}
那么既然又需要将类加载在ioc容器中,又要声明为@ResponseBody。是否能够再进一步简化注解呢?答案是有的。
注解:@ResController = @Controller + @ResponseBody
通过观看源码可以看到
如果在项目下,在WEB-INF外有一个图片,想要通过访问地址去查找。
这里发现通过地址无法进行查找对应的资源,这是为什么呢?
其实是因为在访问时候DispactherServlet老板 会向HandlerMapping秘书询问是否项目下存在这个资源。而HandlerMapping只能查找Handler方法。即那些RequestMapping("路径")的这些对应的方法。而没办法找静态的资源。所以会返回给DIspatcherServlet一个404.
那么应该如何设置才能让DispatcherServlet能够正常获取到静态资源呢? 那就需要用到配置类中继承方法覆写。
那么这个方法的原理是什么呢?为什么这个方法能够访问到静态资源。在找寻到对应源码后如下:
可以看到,其实原理就是用到前面页面跳转的转发功能,而转发可以访问到项目下的资源。
在配置访问静态资源后之后,这时候访问一个资源的流程变成了
DispatcherServlet首先会询问HandlerMapping是否有对应的资源,如果没有就会去访问DefaultServletHandler是否有对应的路径,如果都没有就会显示404.