后端工程化 | SpringBoot 知识点

文章目录

    • [SpringBoot] 后端工程化
      • 1 需求
      • 2 开发流程
      • 3 RequestController 类(操作类)
        • 3.1 简单参数(形参名和请求参数名一致)
        • 3.2 简单参数(形参名和请求参数名不一致)
        • 3.3 复杂实体参数
        • 3.4 数组参数
        • 3.5 集合参数
        • 3.6 时间参数
        • 3.7 json参数
        • 3.8 单个路径参数
        • 3.9 多个路径参数
      • 4 RequestController 类(查询类)
        • 4.1 @ResponseBody
        • 4.2 统一响应结果
      • 5 分层解耦
        • 5.1 三层架构
        • 5.2 分层目录及代码
          • 5.2.1 数据访问层
          • 5.2.2 业务逻辑层
          • 5.2.3 控制层
        • 5.3 分层解耦
          • 5.3.1 耦合问题
          • 5.3.2 解耦思路
        • 5.4 IOC & DI
          • 5.4.1 Bean的声明(IOC)
          • 5.4.2 组件扫描(IOC)
          • 5.4.3 DI


[SpringBoot] 后端工程化

1 需求

  • 需求:基于SpringBoot的方式开发一个web应用,浏览器发起请求“/hello”后,给浏览器返回字符串 “Hello World ”。

SpringBoot通信流程-2023-10-2917:12:43.png

2 开发流程

  • 创建SpringBoot工程项目

第一步:用 IDEA 基于Spring官方骨架,创建SpringBoot工程。

SpringBoot工程骨架-2023-10-2917:15:16.png

第二步:选择 SpringBoot 框架版本,勾选web开发相关依赖。

SpringBoot 框架版本-2023-10-2917:17:41.png

  • 定义RequestController类,添加方法hello,并添加注解

在主包名下,创建 controller 包 + pojo 包

SpringBoot项目的架构-2023-10-2917:23:43.png

package com.itheima.controller;
import org.springframework.web.bind.annotation.*;@RestController
public class RequestController {@RequestMapping("/hello")public String hello(){System.out.println("Hello World ~");return "Hello World !";} 
} 
  • 测试运行

运行 SprintBootWebObjectApplication

3 RequestController 类(操作类)

  • 作用:接收前端的表单数据
3.1 简单参数(形参名和请求参数名一致)
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/simpleParam?name=lisi&age=45
  • 后端
@RestController
public class RequestController {// 第1个请求参数: name=Tom   参数名:name,参数值:Tom// 第2个请求参数: age=10     参数名:age , 参数值:10//springboot方式@RequestMapping("/simpleParam")public String simpleParam(String name , Integer age ){//形参名和请求参数名保持一致System.out.println(name+"  :  "+age);return "OK"; //返回数据}
}
3.2 简单参数(形参名和请求参数名不一致)
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/simpleParam?name=lisi&age=45
  • 后端
@RestController
public class RequestController {// 第1个请求参数: name=Tom   参数名:name,参数值:Tom// 第2个请求参数: age=10     参数名:age , 参数值:10//springboot方式@RequestMapping("/simpleParam")public String simpleParam(@RequestParam(name = "name", required = false) String username , Integer age ){// name:请求参数名,required:参数是否必须System.out.println(username+"  :  "+age);return "OK"; //返回数据}
}
3.3 复杂实体参数
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/complexPojo?username=李四&age=10&address.province=背景&address.city=上海
  • 后端
@RestController
public class RequestController {  //springboot方式@RequestMapping("/complexPojo")public String complexPojo(User user) {System.out.println(user);return "ok";}User{private String username;private int age;private Address address;GetSet 方法;ToString 方法;}Address{private String province;private String city;GetSet 方法;ToString 方法;}
}
3.4 数组参数
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/arrayParam?hobby=string&hobby=java&hobby=python
  • 后端
@RestController
public class RequestController {//springboot方式@RequestMapping("/arrayParam")public String arrayParam(String[] hobby) {System.out.println(Arrays.toString(hobby));return "ok,arrayParam";}
}
3.5 集合参数
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/listParam?hobby=string&hobby=java&hobby=python
  • 后端
@RestController
public class RequestController {//springboot方式@RequestMapping("/listParam")public String listParam(@RequestParam List<String> hobby) {System.out.println(hobby);return "ok,listParam";}
}
3.6 时间参数
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/dateParam?date=2023-02-09 12:08:09
  • 后端
@RestController
public class RequestController {//springboot方式@RequestMapping("/dateParam")public String dateParam(@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") LocalDateTime date) {System.out.println(date);return "ok,dateParam";}
}
3.7 json参数
  • 前端
- 请求方式:POST
- URL: http://localhost:8080/jsonParam
- 请求体:
{"username": "lisi","age": 23,"address": {"province": "beijing","city": "上海"}
}
  • 后端
@RestController
public class RequestController {// 第1个请求参数: name=Tom   参数名:name,参数值:Tom// 第2个请求参数: age=10     参数名:age , 参数值:10//springboot方式@RequestMapping("/jsonParam")public String jsonParam(@RequestBody User user) {System.out.println(user);return "ok,jsonParam";}
}
3.8 单个路径参数
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/path/1000
  • 后端
@RestController
public class RequestController {//springboot方式@RequestMapping("/path/{id}")public String pathParam(@PathVariable Integer id) {System.out.println(id);return "ok,pathParam";}
}
3.9 多个路径参数
  • 前端
- 请求方式:GET
- URL: http://localhost:8080/path/1000/Lisi
  • 后端
@RestController
public class RequestController {//springboot方式@RequestMapping("/path/{id}/{name}")public String pathParam(@PathVariable Integer id, @PathVariable String name) {System.out.println(id);System.out.println(name);return "ok,pathParam";}
}

4 RequestController 类(查询类)

4.1 @ResponseBody
  • 由于 @RestController = @Controller + @ResponseBody ,所以不用重复添加。
4.2 统一响应结果
  • 定义在一个实体类 Result 来包含以上信息。代码如下:
public class Result {private Integer code;//响应码,1 代表成功; 0 代表失败private String msg;  //响应码 描述字符串private Object data; //返回的数据public Result() { }public Result(Integer code, String msg, Object data) {this.code = code;this.msg = msg;this.data = data;}public Integer getCode() {return code;}public void setCode(Integer code) {this.code = code;}public String getMsg() {return msg;}public void setMsg(String msg) {this.msg = msg;}public Object getData() {return data;}public void setData(Object data) {this.data = data;}//增删改 成功响应(不需要给前端返回数据)public static Result success(){return new Result(1,"success",null);}//查询 成功响应(把查询结果做为返回数据响应给前端)public static Result success(Object data){return new Result(1,"success",data);}//失败响应public static Result error(String msg){return new Result(0,msg,null);}
}

5 分层解耦

5.1 三层架构
  • 请求处理、响应数据(Controller):控制层。负责接收页面的请求,给页面响应数据。
  • 逻辑处理(Service):业务逻辑层。负责业务逻辑处理的代码。
  • 数据访问(Dao):数据访问层(Data Access Object),也称为持久层。负责业务数据的维护操作,包括增、删、改、查等操作。

在这里插入图片描述

5.2 分层目录及代码

1698590017953-2023-10-2922:33:38.png

5.2.1 数据访问层
  • 负责数据的访问操作,包含数据的增、删、改、查

  • 数据访问接口

//数据访问层接口(制定标准)
public interface EmpDao {//获取员工列表数据public List<Emp> listEmp();
}
  • 数据访问实现类
//数据访问实现类
public class EmpDaoA implements EmpDao {@Overridepublic List<Emp> listEmp() {//1. 加载并解析emp.xmlString file = this.getClass().getClassLoader().getResource("emp.xml").getFile();System.out.println(file);List<Emp> empList = XmlParserUtils.parse(file, Emp.class);return empList;}
}
5.2.2 业务逻辑层
  • 处理具体的业务逻辑

  • 业务接口

//业务逻辑接口(制定业务标准)
public interface EmpService {//获取员工列表public List<Emp> listEmp();
}
  • 业务实现类
//业务逻辑实现类(按照业务标准实现)
public class EmpServiceA implements EmpService {//dao层对象private EmpDao empDao = new EmpDaoA();@Overridepublic List<Emp> listEmp() {//1. 调用dao, 获取数据List<Emp> empList = empDao.listEmp();//2. 对数据进行转换处理 - gender, jobempList.stream().forEach(emp -> {//处理 gender 1: 男, 2: 女String gender = emp.getGender();if("1".equals(gender)){emp.setGender("男");}else if("2".equals(gender)){emp.setGender("女");}//处理job - 1: 讲师, 2: 班主任 , 3: 就业指导String job = emp.getJob();if("1".equals(job)){emp.setJob("讲师");}else if("2".equals(job)){emp.setJob("班主任");}else if("3".equals(job)){emp.setJob("就业指导");}});return empList;}
}
5.2.3 控制层
  • 接收前端发送的请求,对请求进行处理,并响应数据
@RestController
public class EmpController {//业务层对象private EmpService empService = new EmpServiceA();@RequestMapping("/listEmp")public Result list(){//1. 调用service层, 获取数据List<Emp> empList = empService.listEmp();//3. 响应数据return Result.success(empList);}
}
5.3 分层解耦
5.3.1 耦合问题
  • 内聚:软件中各个功能模块内部的功能联系。

  • 耦合:衡量软件中各个层/模块之间的依赖、关联的程度。

软件设计原则:高内聚低耦合。

高内聚:一个模块中的代码,各个代码块之间关系越紧密,则内聚性越高。

低耦合:软件中各个层、模块之间的依赖(相互关联)性越低,则耦合度越低。

高内聚、低耦合-2023-11-621:24:47.png

5.3.2 解耦思路
  • 解决思路

    1. 提供一个容器,容器中存储一些对象(例:EmpService对象)
    2. controller程序从容器中获取EmpService类型的对象
  • 解耦操作

    • 控制反转:[ Inversion Of Control ],简称IOC。对象的创建控制权由程序自身转移到外部(容器),这种思想称为控制反转。

      对象的创建权由程序员主动创建转移到容器(由容器创建、管理对象)。这个容器称为:IOC容器或Spring容器

    • 依赖注入:[ Dependency Injection ],简称DI。容器为应用程序提供运行时,所依赖的资源,称之为依赖注入。

      程序运行时需要某个资源,此时容器就为其提供这个资源。

      例:EmpController程序运行时需要EmpService对象,Spring容器就为其提供并注入EmpService对象

5.4 IOC & DI
5.4.1 Bean的声明(IOC)
  • ICO 容器管理:IOC控制反转,就是将对象的控制权交给Spring的IOC容器,由IOC容器创建及管理对象。IOC容器创建的对象称为bean对象。

把某个对象交给IOC容器管理,需要在类上添加一个注解:@Component

注解说明位置
@Component声明bean的基础注解全部类通用注解(除控制类)
@Controller@Component的衍生注解标注在控制器类上
@Service@Component的衍生注解标注在业务逻辑类上
@Repository@Component的衍生注解标注在数据访问类上(由于与mybatis整合,用的少)

  • @RestController = @Controller + @ResponseBody
  • 声明bean的时候,可以通过value属性指定bean的名字,如果没有指定,默认为类名首字母小写。
  • 使用以上四个注解都可以声明bean,但是在springboot集成web开发中,声明控制器bean只能用@Controller。
5.4.2 组件扫描(IOC)
  • Bean 想要生效,还需要被组件扫描。
// 使用四大注解声明的 Bean,要想生效,还需要被组件扫描注解 @ComponentScan 扫描
@ComponentScan("dao", "com.Jiacheng")
@SpringBootApplication
public class SprintBootWebObjectApplication {public static void main(String[] args) {SpringApplication.run(SprintBootWebObjectApplication.class, args);}
}

:不推荐组件扫描,按照 SprintBoot 项目结构创建目录即可

5.4.3 DI
  • 依赖注入:IOC容器要为应用程序去提供运行时所依赖的资源,而资源指的就是对象。
// 依赖注入:@Autowired注解,按照类型进行自动装配。
@Autowired
private EmpService empService;
  • 依赖冲突:在 IOC 容器中,存在多个相同类型的 Bean 对象,程序运行会报错。
注解说明位置
@Autowired按照类型自动装配 Bean 对象类内
@Primary让当前类的 Bean 生效类上
@Qualifier(“serviceA”)指定注入的 Bean 对象(搭配@Autowired)类内
@Resource(name = “serviceB”)按照 Bean的名称进行注入类内
// 1.@Autowired 注入
@Autowired
private EmpService empService;// 2.@Primary 注入
@Primary
public class EmpServiceA implements EmpService {}// 3.@Qualifier() 注入,指定当前要注入的bean对象
@Autowired
@Qualifier("empServiceA")
private EmpService empService;// 4.@Resource() 注入
@Resource(name = "empServiceB")
private EmpService empService;
  • 面试题 : @Autowird 与 @Resource的区别?
  1. @Autowired 是spring框架提供的注解,而@Resource是JDK提供的注
  2. @Autowired 默认是按照类型注入,而@Resource是按照名称注入

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/132876.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

COOHOM通过采用亚马逊云科“专库专用”的方式,为云原生的构建提供稳定的数据支撑

全球化浪潮下&#xff0c;面对全球化业务发展带来的新需求与新挑战&#xff0c;越来越多的企业开启了云原生构建旅程&#xff0c;以推动业务系统快速迭代&#xff0c;为国际业务的拓展打下坚实的基础。COOHOM是杭州群核信息技术有限公司旗下的国际化品牌。为全球企业和个人提供…

《面向对象软件工程》笔记——1-2章

“学习不仅是一种必要&#xff0c;而且是一种愉快的活动。” - 尼尔阿姆斯特朗 文章目录 第一章 面向对象软件工程的范畴历史方面经济方面维护方面现代软件维护观点交付后维护的重要性 需求、分析和设计方面团队开发方面没有计划&#xff0c;测试&#xff0c;文档阶段的原因面向…

8-3、T型加减速单片机程序【51单片机控制步进电机-TB6600系列】

摘要&#xff1a;根据前两节内容&#xff0c;已完成所有计算工作&#xff0c;本节内容介绍具体单片机程序流程及代码 一、程序流程图 根据前两节文章内容可知&#xff0c;T型加减速的关键内容是运动类型的判断以及定时器初值的计算&#xff0c;在输出运动参数后即可判断出运动…

Oracle(13)Maintaining Data Integrity

目录 一、基础知识 1、Data Integrity 数据库的完整性 2、Types of Constraints 约束类型 3、Constraint States 约束状态 4、Guidelines for Constraints 约束准则 二、基础操作 1、Enabling Constraints 启用约束 2、命令方式创建约束 3、修改表创建的约束 4、删除约…

JavaScript_Element对象_方法

1、Element.focus() Element.focus方法用于将当前页面的焦点&#xff0c;转移到指定元素上 2、Element.blur() Element.blur方法用于将焦点从当前元素移除 3、Element.remove() Element.remove方法用于将当前元素节点从它的父节点移除 4、Element.getBoundingClientRect() …

常用的电子邮件服务提供商有哪些?

当我们讨论常用的电子邮件服务时&#xff0c;可以根据国内和国外进行分类观察。以下是一些常见的国内和国外电子邮件服务。 什么是国外邮箱和国内邮箱&#xff1f; 国外邮箱是指在国外注册和使用的电子邮箱&#xff0c;而国内邮箱则是在国内注册和使用的电子邮箱。 国外邮箱是指…

链表(1)

目录 单链表 主函数test.c test1 test2 test3 test4 头文件&函数声明SList.h 函数实现SList.c 打印SLPrint 创建节点CreateNode 尾插SLPushBack 头插SLPushFront 头删SLPopBck 尾删SLPopFront 易错点 本篇开始链表学习。今天主要是单链表&OJ题目。 单链…

接口请求断言

接口请求断言是指在发起请求之后&#xff0c;对返回的响应内容去做判断&#xff0c;用来查看是否响应内容是否与规定的返回值相符。 在发起请求后&#xff0c;我们使用一个变量 r 存储响应的内容&#xff0c;也就是 Response 对象。 Response 对象有很多功能强大的方法可以调…

城市内涝怎么预警?万宾科技内涝积水监测仪

在城市运行过程中&#xff0c;城市内涝问题频繁出现&#xff0c;影响城市管理水平的提升&#xff0c;也会进一步减缓城市基础设施建设。尤其近几年来&#xff0c;城市内涝灾害频繁出现&#xff0c;在沿海地区内涝所带来的安全隐患成为城市应急管理部门的心头大患。城市内涝的背…

【JVM系列】- 挖掘·JVM堆内存结构

挖掘JVM堆内存结构 文章目录 挖掘JVM堆内存结构堆的核心概念堆的特点 堆的内存结构内存划分新生代/新生区&#xff08;Young Generation&#xff09;老年代&#xff08;Tenured Generation&#xff09;永久代&#xff08;或元数据区&#xff09;&#xff08;PermGen 或 MetaSpa…

2023.11.4 Idea 配置国内 Maven 源

目录 配置国内 Maven 源 重新下载 jar 包 配置国内 Maven 源 <mirror><id>alimaven</id><name>aliyun maven</name><url>http://maven.aliyun.com/nexus/content/groups/public/</url><mirrorOf>central</mirrorOf> …

ChatGPT对未来发展的影响?一般什么时候用到GPT

ChatGPT以其强大的自然语言处理能力对未来的发展具有重要影响。以下是ChatGPT的潜在影响和一般使用情况&#xff1a; 改善自然语言理解和生成&#xff1a;ChatGPT和类似的模型可以改善机器对人类语言的理解和生成。这将有助于改进各种应用领域&#xff0c;包括智能助手、聊天机…

HMM与LTP词性标注之命名实体识别与HMM

文章目录 知识图谱介绍NLP应用场景知识图谱&#xff08;Neo4j演示&#xff09;命名实体识别模型架构讲解HMM与CRFHMM五大要素&#xff08;两大状态与三大概率&#xff09;HMM案例分享HMM实体识别应用场景代码实现 知识图谱介绍 NLP应用场景 图谱的本质&#xff0c;就是把自然…

Linux友人帐之网络编程基础www服务器

一、概述 1.1www基础 WWW&#xff08;World Wide Web&#xff0c;万维网&#xff09;是一种分布式、全球性的信息服务系统&#xff0c;是集成Internet、Web浏览器和Web服务器等技术而形成的一个庞大的、涉及全球的信息网络。 用户在浏览器中输入www.cqvie.edu.cn访问该网站主页…

day2 ARM基础

.text .globl _start _start:mov r0,#0 mov r1,#0 addfunc:add r0,r0,#1 r0自增1adds r1,r1,r0 R1实现1~100累加cmp r0,#100 判断r0是否到100bleq loop r0等于100 进入死循环 blne addfunc r0等于100跳转至循环累加 loop:b loopstop:b stop.end 【汇编…

同样是PM,产品经理和项目经理有啥不一样?

大家好&#xff0c;我是老原。身边有很多人都问&#xff1a; “干几年的技术可以做到项目经理&#xff1f;” “我要从项目经理转型到产品经理吗&#xff1f;” “产品经理和项目经理&#xff0c;哪个发展前&#xff08;钱&#xff09;景更好” …… 不难发现&#xff0c;…

Microsoft Dynamics 365 CE 扩展定制 - 7. 安全

在本章中,我们将介绍以下内容: 构建累积安全角色配置业务单元层次结构基于分层位置配置访问配置和分配字段级安全组建团队并共享设置访问团队对静止数据进行加密以满足FIPS 140-2标准管理Dynamics 365在线SQLTDE加密密钥简介 Dynamics 365是一个强大的平台,具有超过10年的良…

python数据结构和算法基础(第一节,数据结构和算法基础)

01.算法引入 02.时间复杂度和大o表示法 2.1时间复杂度与大o表示法 引入&#xff1a; 2.1时间复杂度与大o表示法 pycharm中&#xff0c;快速多行注释&#xff0c;ctrl/ 2.2最坏时间复杂度与计算规则 2.3常见时间复杂度与大小关系 03.python列表和字典 3.1代码执行时间测量木…

Git 内容学习

一、Git 的理解 Git是一个分布式版本控制系统&#xff08;Distributed Version Control System&#xff0c;简称 DVCS&#xff09;&#xff0c;用于对项目源代码进行管理和跟踪变更。分为两种类型的仓库&#xff1a;本地仓库和远程仓库。 二、Git 的工作流程 详解如下&#x…

webgoat-Broken Access ControlI 访问控制失效

Insecure Direct Object References 直接对象引用 直接对象引用是指应用程序使用客户端提供的输入来访问数据和对象。 例子 使用 GET 方法的直接对象引用示例可能如下所示 https://some.company.tld/dor?id12345 https://some.company.tld/images?img12345 https://some.…