【javaweb】学习日记Day5 - 请求响应 分层解耦 IOC DI 三层架构

news2024/12/24 2:40:58

目录

一、请求响应

1、请求

(1)简单参数

① GET请求

 ② POST请求

③ 假如形参与请求参数不一致

(2)实体参数

① 简单实体对象

② 复杂实体对象

(3)数组参数

(4)集合参数

(5)日期参数

(6) json参数

 (7)路径参数

 2、统一响应

(1)设置一个响应类Result

(2)设置响应代码

3、案例

(1) 引入相关资料

(2)编写程序处理请求,响应数据

 (3)测试

二、分层解耦

1、三层架构

(1)编写Dao层代码

① Dao层接口 

② Dao层接口实现类 

(2)编写service层代码

① Service层接口  

② Service层接口实现类 

(3)编写controller层代码

2、分层解耦

(1)概念

 (2)控制反转与依赖注入

3、IOC和DI

(1)Controller层

(2)Service层

(3)Dao层

(4)更改同层Bean对象

(5)如何看查Bean对象?

4、IOC详解

5、DI详解 

(1)@Primary

(2)@Autowired+@Qualifier("bean的名称")

(3)@Resource(name="bean的名称")


一、请求响应

1、请求

(1)简单参数

简单参数:你在方法里写的形参请求参数名称匹配,则返回方法内数据

package com.roye;

import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController  //请求处理类
public class Springbootwebrequest {
    //基于springboot方式
    @RequestMapping("/example01")  //浏览器将来请求/example01这个地址,会调用该方法
    public static String example01(String name,Integer age)
    {
        System.out.println("name:"+name+" age:"+age);
        return "OK";
    }
}

① GET请求

运行后用postman新建一个get请求,输入网页地址及参数即可获取内容

 

 ② POST请求

post内容在请求体中

③ 假如形参与请求参数不一致

可以用@RequestParam完成映射

@RestController  //请求处理类
public class Springbootwebrequest {
    //基于springboot方式
    @RequestMapping("/example01")  //浏览器将来请求/example01这个地址,会调用该方法
    public static String example01(@RequestParam(name="name") String username, Integer age)
    {
        System.out.println("name:"+username+" age:"+age);
        return "OK";
    }
}

其中required默认值为true,意思为请求参数必须传递

@RestController  //请求处理类
public class Springbootwebrequest {
    //基于springboot方式
    @RequestMapping("/example01")  //浏览器将来请求/example01这个地址,会调用该方法
    public static String example01(@RequestParam(name="name",required = false) String username, Integer age)
    {
        System.out.println("name:"+username+" age:"+age);
        return "OK";
    }
}

(2)实体参数

规则:请求参数名与形参名相同

① 简单实体对象

先定义一个类

public class User {
    private String name;
    private String age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age='" + age + '\'' +
                '}';
    }
}

方法中参数设为类

//实体参数-简单实体
    @RequestMapping("/example02")
    public static String example02(User x) {
        System.out.println(x);
        return "OK";
    }

② 复杂实体对象

//实体参数-复杂实体
    @RequestMapping("/example03")
    public static String example03(User x) {
        System.out.println(x);
        return "OK";

就是类嵌套着类! 

public class User {
    private String name;
    private String age;
    private Address address;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }

    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age='" + age + '\'' +
                ", address='" + address + '\'' +
                '}';
    }
}
public class Address {
    private String province;
    private String city;

    public String getProvince() {
        return province;
    }

    public void setProvince(String province) {
        this.province = province;
    }

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

    @Override
    public String toString() {
        return "Address{" +
                "province='" + province + '\'' +
                ", city='" + city + '\'' +
                '}';
    }
}

(3)数组参数

    //数组参数
    @RequestMapping("/example04")
    public static String example04(String[] hobby) {
        System.out.println(Arrays.toString(hobby));
        return "OK";
    }

 

(4)集合参数

    //集合参数
    @RequestMapping("/example05")
    public static String example05(@RequestParam List<String> hobby) {
        System.out.println(hobby);
        return "OK";
    }

(5)日期参数

    //日期时间参数
    @RequestMapping("/example06")
    public static String example06(@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") LocalDateTime time) {
        System.out.println(time);
        return "OK";
    }

 

(6) json参数

    //json参数
    @RequestMapping("/example07")
    public static String example07(@RequestBody User x) {
        System.out.println(x);
        return "OK";
    }

 

 (7)路径参数

    //路径参数
    @RequestMapping("/path/{id}/{name}")
    public static String example08(@PathVariable Integer id,@PathVariable String name) {
        System.out.println(id+":"+name);
        return "OK";
    }

 

 2、统一响应

@RestControlled 包含 @ResponseBody

(1)设置一个响应类Result

package com.roye.day5;

/**
 * 统一响应结果封装类
 */
public class Result {
    private Integer code ;//1 成功 , 0 失败
    private String msg; //提示信息
    private Object data; //数据 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(Object data){
        return new Result(1, "success", data);
    }
    public static Result success(){
        return new Result(1, "success", null);
    }
    public static Result error(String msg){
        return new Result(0, msg, null);
    }

    @Override
    public String toString() {
        return "Result{" +
                "code=" + code +
                ", msg='" + msg + '\'' +
                ", data=" + data +
                '}';
    }
}

(2)设置响应代码

    @RequestMapping("/hello")
    public Result hello(){
        System.out.println("Hello World ~");
        //return new Result(1,"success","Hello World ~");
        return Result.success("Hello World ~");
    }

    @RequestMapping("/getAddr")
    public Result getAddr(){
        Address addr = new Address();
        addr.setProvince("广东");
        addr.setCity("深圳");
        return Result.success(addr);
    }

    @RequestMapping("/listAddr")
    public Result listAddr(){
        List<Address> list = new ArrayList<>();

        Address addr = new Address();
        addr.setProvince("广东");
        addr.setCity("深圳");

        Address addr2 = new Address();
        addr2.setProvince("陕西");
        addr2.setCity("西安");

        list.add(addr);
        list.add(addr2);
        return Result.success(list);
    }
}

 

3、案例

(1) 引入相关资料

 

 Springboot项目的静态资源(HTML、css、js等前端)默认存放目录 static

(2)编写程序处理请求,响应数据

package com.roye;

import com.roye.day5.Emp;
import com.roye.day5.Result;
import com.roye.tools.XmlParserUtils;
import org.springframework.context.annotation.EnableMBeanExport;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

import static org.springframework.util.ResourceUtils.getFile;

@RestController
public class EmpController {

    @RequestMapping("/listEmp")
    public Result list(){
        //1、加载并解析emp.xml
        String file = this.getClass().getClassLoader().getResource("emp.xml").getFile();
        System.out.println(file);
        List<Emp> emplist = XmlParserUtils.parse(file, Emp.class);

        //2、对数据进行转换处理
        emplist.stream().forEach(x -> {
            //处理gender 1:男 2:女
            String gender=x.getGender();
            if("1".equals(gender)) x.setGender("男");
            else if("2".equals(gender)) x.setGender("女");

            //处理job 1:讲师 2:班主任 3:就业指导
            String job=x.getJob();
            if("1".equals(job)) x.setJob("讲师");
            else if("2".equals(job)) x.setJob("班主任");
            else if("3".equals(job)) x.setJob("就业指导");
        });

        //3、响应数据
        return Result.success(emplist);
    }
}

 (3)测试

 

二、分层解耦

1、三层架构

对于上面那个案例,我们把代码全部写一块,复用性差,难以维护

  • controller:控制层,接收前端发送的请求,对请求进行处理,并响应数据
  • service:业务逻辑层,处理具体业务逻辑
  • dao:数据访问层,包括数据的增删改查

(1)编写Dao层代码

① Dao层接口 

//接口
package com.roye.dao;

import com.roye.day5.Emp;

import java.util.List;

public interface EmpDao {
    //获取员工列表数据
    public List<Emp> listEmp();
}

② Dao层接口实现类 

//接口实现类
package com.roye.dao.impl;

import com.roye.dao.EmpDao;
import com.roye.day5.Emp;
import com.roye.tools.XmlParserUtils;

import java.util.List;

public class EmpDaoA implements EmpDao {

    @Override
    public List<Emp> listEmp() {
        //1、加载并解析emp.xml
        String file = this.getClass().getClassLoader().getResource("emp.xml").getFile();
        System.out.println(file);
        List<Emp> emplist = XmlParserUtils.parse(file, Emp.class);
        return emplist;
    }
}

(2)编写service层代码

① Service层接口  

package com.roye.service;

import com.roye.day5.Emp;

import java.util.List;

public interface EmpService {
    //获取员工列表
    public List<Emp> listEmp();
}

② Service层接口实现类 

package com.roye.service.impl;

import com.roye.dao.EmpDao;
import com.roye.dao.impl.EmpDaoA;
import com.roye.day5.Emp;

import java.util.List;

public class EmpService implements com.roye.service.EmpService {
    private EmpDao empDao=new EmpDaoA();  //设接口实现类
    @Override
    public List<Emp> listEmp() {
        //1、调用Dao,获取数据
        List<Emp> empList=empDao.listEmp();

        //2、对数据进行转换处理
        empList.stream().forEach(x -> {
            //处理gender 1:男 2:女
            String gender=x.getGender();
            if("1".equals(gender)) x.setGender("男");
            else if("2".equals(gender)) x.setGender("女");

            //处理job 1:讲师 2:班主任 3:就业指导
            String job=x.getJob();
            if("1".equals(job)) x.setJob("讲师");
            else if("2".equals(job)) x.setJob("班主任");
            else if("3".equals(job)) x.setJob("就业指导");
        });
        return empList;
    }
}

(3)编写controller层代码

package com.roye.controller;

import com.roye.day5.Emp;
import com.roye.day5.Result;
import com.roye.service.EmpService;
import com.roye.tools.XmlParserUtils;
import org.springframework.context.annotation.EnableMBeanExport;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

import static org.springframework.util.ResourceUtils.getFile;

@RestController
public class EmpController {
    private EmpService empService=new com.roye.service.impl.EmpService();
    
    @RequestMapping("/listEmp")
    public Result list(){
        //1、调用Service,获取数据
        List<Emp> empList=empService.listEmp();
        
        //2、响应数据
        return Result.success(empList);
    }
}

2、分层解耦

(1)概念

  • 内聚:每个模块间功能的联系
  • 耦合:衡量软件中各模块间的依赖关联程度
  • 软件设计原则:高内聚低耦合

 (2)控制反转与依赖注入

  • 控制反转 IOC:对象创建控制权由程序内部移向外部(容器)
  • 依赖注入 DI:容器为程序运行所依赖的资源
  • Bean对象:IOC容器中创建、管理的对象

3、IOC和DI

注解在各层的应用
注解作用Controller层Service层Dao层
@Component

声明bean对象,将当前类交给IOC容器管理

@Autowired运行时,IOC容器会提供该类型的Bean对象,并赋值给该变量
  • Controller层需要调用Service层(进行逻辑处理)【@Autowired】,因此将Service层放入IOC容器【@Component】
  • Service层需要调用Dao层(进行获取数据)【@Autowired】,因此将Dao层放入IOC容器【@Component】

(1)Controller层

@RestController
public class EmpController {
    @Autowired  //运行时,IOC容器会提供该类型的Bean对象,并赋值给该变量
    private EmpService empService;

    @RequestMapping("/listEmp")
    public Result list(){
        //1、调用Service,获取数据
        List<Emp> empList=empService.listEmp();

        //2、响应数据
        return Result.success(empList);
    }
}

(2)Service层

@Component  //将当前类交给IOC容器管理,称为IOC容器中的Bean
public class EmpService implements com.roye.service.EmpService {
    @Autowired  //运行时,IOC容器会提供该类型的Bean对象,并赋值给该变量
    private EmpDao empDao;

    @Override
    public List<Emp> listEmp() {
        //1、调用Dao,获取数据
        List<Emp> empList=empDao.listEmp();

        //2、对数据进行转换处理
        empList.stream().forEach(x -> {
            //处理gender 1:男 2:女
            String gender=x.getGender();
            if("1".equals(gender)) x.setGender("男");
            else if("2".equals(gender)) x.setGender("女");

            //处理job 1:讲师 2:班主任 3:就业指导
            String job=x.getJob();
            if("1".equals(job)) x.setJob("讲师");
            else if("2".equals(job)) x.setJob("班主任");
            else if("3".equals(job)) x.setJob("就业指导");
        });
        return empList;
    }
}

(3)Dao层

@Component  //将当前类交给IOC容器管理,称为IOC容器中的Bean
public class EmpDaoA implements EmpDao {

    @Override
    public List<Emp> listEmp() {
        //1、加载并解析emp.xml
        String file = this.getClass().getClassLoader().getResource("emp.xml").getFile();
        System.out.println(file);
        List<Emp> emplist = XmlParserUtils.parse(file, Emp.class);
        return emplist;
    }
}

(4)更改同层Bean对象

假设Service层出现两个类

 要将EmpServiceB应用,则把EmpService中的@Component注释掉

(5)如何看查Bean对象?

4、IOC详解

  •  前面声明bean四大注解,想要生效,需要被组件扫描注解@ComponentScan扫描
  • @ComponentScan已经包含在启动类声明注解@SpringBootApplication中,默认扫描范围为启动类所在包及子包

 

5、DI详解 

@Autowired注解,默认是按类型注解,如果存在多个相同类型的bean,将会报错!

(1)@Primary

(2)@Autowired+@Qualifier("bean的名称")

(3)@Resource(name="bean的名称")

 

 

 

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

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

相关文章

研究生定向培养学徒对象及说明

研究生定向培养学徒开始招募啦&#xff0c;招募对象可以 1、免费学习 2、全真企业项目实战 3、拥有就业推荐机会 4、提供副业机会 研究生定向培养学徒报名时间&#xff1a; 2023年8月22日-2023年9月10日 研究生定向培养学徒招募对象&#xff1a; 1.毕业年度研究…

如何在地平线J5上部署RTA-VRTE v2.2应用程序

在地平线J5上部署RTA-VRTE v2.2应用程序流程图: 虽然在J5上使用ifconfig 命令看不到can0和can1被启动 登陆系统后ifconfig -a仍然能看到can0和can1。

Python(八十四)字符串的切片操作

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…

Linux TCP协议——三次握手,四次挥手

一、TCP协议介绍 TCP协议是可靠的、面向连接的、基于字节流的传输层通信协议。 TCP的头部结构&#xff1a; 源/目的端口号: 表示数据是从哪个进程来, 到哪个进程去;&#xff08;tcp是传输层的协议&#xff0c;端与端之间的数据传输&#xff0c;在TCP和UDP协议当中不会体现出I…

基于paddleocr的文档识别

1、版面分析 使用轻量模型PP-PicoDet检测模型实现版面各种类别的检测。 数据集&#xff1a; 英文&#xff1a;publaynet数据集的训练集合中包含35万张图像&#xff0c;验证集合中包含1.1万张图像。总共包含5个类别。 中文&#xff1a;CDLA据集的训练集合中包含5000张图像&a…

Vue3:通过路由写多个页面,通过不同的路径可以进入不同的页面

前言 Vue3&#xff1a;想通过路由写2个页面&#xff0c;不同的路径可以进入不同的页面 实现步骤 1、创建Vue3项目 通过脚手架创建一个Vue3的项目&#xff0c;然后在此基础上对文件进行增删改&#xff0c;修改成自己需要的项目框架 2、views文件夹 对应 页面文件 如果需要…

网络编程——网络基础知识

目录 一、网络历史两个重要名词1.1 阿帕网1.2 TCP/IP协议 二、局域网和广域网三、IP地址3.1 基本概念3.2 划分(IPV4)3.3 特殊IP地址3.4 子网掩码3.5 重新组网 四、网络模型4.1 网络的体系结构&#xff1a;4.2 OSI与TCP/IP模型4.2.1 OSI模型4.2.2 TCP/IP模型4.2.3 OSI和TCP/IP模…

C++,类的特殊函数练习

设计一个Per类&#xff0c;类中包含私有成员:姓名、年龄、指针成员身高、体重&#xff0c;再设计一个Stu类&#xff0c;类中包含私有成员:成绩、Per类对象p1&#xff0c;设计这两个类的构造函数、析构函数和拷贝构造函数。 #include <iostream> using namespace std;cla…

OpenAI的Superalignment策略:计算为王

卷友们好&#xff0c;我是rumor。 对于怎么实现AGI这个玄学的目标&#xff0c;感觉大家都是差不多的状态&#xff1a;咱也不知道怎么做&#xff0c;但就是觉得现在的LLM技术还远远不够。 所以之前看到OpenAI说要用模型去做对齐研究[1]&#xff0c;以及最近发话要4年内做出Super…

创建R包-2.1:在RStudio中使用Rcpp制作R-Package(更新于2023.8.23)

目录 0-前言 1-在RStudio中创建R包项目 2-创建R包 2.1通过R函数创建新包 2.2在RStudio通过菜单来创建一个新包 2.3关于R包创建的说明 3-添加R自定义函数 4-添加C函数 0-前言 目标&#xff1a;在RStudio中创建一个R包&#xff0c;这个R包中包含C函数&#xff0c;接口是Rc…

牛客复盘] 2023河南萌新联赛第(七)场:信息工程大学 B\I 20230823

牛客复盘] 2023河南萌新联赛第&#xff08;七&#xff09;场&#xff1a;信息工程大学 B\I 20230823 总结B 七夕1. 题目描述2. 思路分析3. 代码实现 I 细胞分裂1. 题目描述2. 思路分析3. 代码实现 六、参考链接 总结 场外OB做了B和I题&#xff0c;只能说这场有点离谱。B 并查…

律师事务所微信小程序开发方案:实现智能化服务与用户体验的完美结合

随着移动互联网的快速发展&#xff0c;微信成为了人们日常生活中不可或缺的社交工具。为了满足用户对便捷、高效法律服务的需求&#xff0c;律师事务所微信小程序应运而生。本文将探讨律师事务所微信小程序的开发方案&#xff0c;旨在为读者提供一个专业、思考深度和逻辑性的指…

黑客自学笔记

谈起黑客&#xff0c;可能各位都会想到&#xff1a;盗号&#xff0c;其实不尽然&#xff1b;黑客是一群喜爱研究技术的群体&#xff0c;在黑客圈中&#xff0c;一般分为三大圈&#xff1a;娱乐圈 技术圈 职业圈。 娱乐圈&#xff1a;主要是初中生和高中生较多&#xff0c;玩网恋…

【RHEL】硬盘分区与格式化

fdisk命令 在linux中&#xff0c;fdisk是基于菜单的命令。对硬盘分区时&#xff0c;可以在fdisk命令后面直接加上要分区的硬盘作为参数(分区工具) 利用如下所示命令&#xff0c;打开fdisk操作菜单。 输入p,查看当前分区表。从命令执行结果可以到&#xff0c;/dev/mapper/rhel…

网络拓扑结构

目录 1.网络拓扑结构 1.星型拓扑结构 2.网型拓扑结构 3. 数制介绍 3.1 数制的基本概念 1.网络拓扑结构 网络拓扑结构是指用传输媒体互连各种设备的物理布局&#xff0c;也就是用什么方式连接网络中的计算机、网络设备&#xff0c;它的结构有星型拓扑&#xff0c;总线型拓扑…

JVM 之 垃圾收集算法详解

文章目录 一、标记清楚算法二、标记复制算法三、标记整理算法四、不同垃圾收集算法优缺点总结1. 标记-清除算法&#xff1a;2. 标记-复制算法&#xff1a;3. 标记-整理算法&#xff1a; 一、标记清楚算法 标记清除是一种简单而直接的垃圾回收算法。它的执行流程如下&#xff1…

【Rust】Rust学习 第十九章高级特征

现在我们已经学习了 Rust 编程语言中最常用的部分。在第二十章开始另一个新项目之前&#xff0c;让我们聊聊一些总有一天你会遇上的部分内容。你可以将本章作为不经意间遇到未知的内容时的参考。本章将要学习的功能在一些非常特定的场景下很有用处。虽然很少会碰到它们&#xf…

SpringBoot案例-文件上传

目录 简介 文件上传前端页面三要素 服务端接收文件 小结 本地储存 实现 代码优化 小结 阿里云OSS 阿里云 阿里云OSS 使用第三方服务--通用思路 准备工作 参照官方SDK代码&#xff0c;编写入门程序 集成使用 阿里云OSS-使用步骤 阿里云OSS使用步骤 参照SDK编写入…

【QT5-自我学习-线程qThread练习-两种使用方式-2:通过继承Qobject类-自己实现功能函数方式-基础样例】

【QT5-自我学习-线程qThread练习-两种使用方式-2&#xff1a;通过继承Qobject类-自己实现功能函数方式-基础样例】 1、前言2、实验环境3-1、学习链接-参考文章3-2、先前了解-自我总结&#xff08;1&#xff09;线程处理逻辑事件&#xff0c;不能带有主窗口的事件&#xff08;2&…

攻防世界-base÷4

原题 解题思路 base644&#xff0c;莫不是base16&#xff0c;base16解码网站&#xff1a; 千千秀字