简单的springboot项目

传参方式

URL 传参

URL 传参的两种常见方式是通过查询参数和路径参数。

查询参数:

查询参数是通过在 URL 后面使用 ? 字符,然后以 key=value 的形式添加到 URL 中。多个查询参数之间使用 & 符号分隔。例如:https://example.com/api?name=John&age=30。
在前端,可以使用 JavaScript 的 window.location.search 或者 URLSearchParams API 来获取和解析查询参数的值。

路径参数:

路径参数是将参数直接嵌入到 URL 的路径中。通常使用占位符的形式表示,并用实际的值替换占位符部分。例如:https://example.com/api/users/{userId}。
在前端,可以通过路由配置来定义带有参数的路径,并使用框架或库提供的方法来解析路径参数的值。例如,在 Vue.js 中可以使用 Vue Router 的动态路由来处理路径参数。

这两种方式在前端开发中都很常见,具体使用哪种方式取决于你的需求和项目的设计。查询参数适合传递可选的、不敏感的参数,而路径参数适合传递必需的、敏感的参数。

实例

查询参数用@RequestBody接收
在controller层不需要规定路径

@GetMappingpublic Result queryEmpsByPage(@RequestParam("name") String name,@RequestParam(value = "gender", defaultValue = "0") Integer gender,@RequestParam("begin") String begin, @RequestParam("end") String end,@RequestParam(value = "page", defaultValue = "1") Integer page,@RequestParam(value = "pageSize", defaultValue = "5") Integer pageSize) {return empService.queryEmpsByPage(name, gender, begin, end, page, pageSize);}

路径参数用@PathVariable接收
@PathVariable 路径传参
在controller层要规定路径

 @DeleteMapping("/{ids}")public Result delete(@PathVariable List<Integer> ids){empService.delete(ids);return Result.success();}

对应的xml文件代码如下,动态sql

<delete id="deteleByIfs">delete from emp where id in<foreach collection="ids" item="id" separator="," open="(" close=")">#{id}</foreach></delete>

在测试的时候直接把参数拼在路径上
在这里插入图片描述

HTTP 请求体

在 HTTP 请求体中发送附加的数据。这种方式通常用于向后端发送较长或复杂的数据,例如 JSON 数据或文件上传。在前端可以使用 XMLHttpRequest 对象或 fetch API 发送 HTTP 请求,在后端可以使用对应的框架或库解析请求体并获取数据。

@RequestParam和@Param的区别

@RequestParam 和 @Param 是两个不同的注解,它们在不同的框架或库中使用,并有不同的含义和用途。

  1. @RequestParam:
    @RequestParam 注解通常用于后端框架(如Spring MVC)中,用于从 HTTP 请求中获取查询参数或表单字段的值。它可以指定参数的名称、是否是必需的、默认值等属性。在Spring MVC中,可以通过该注解将请求参数绑定到方法的参数上。

    示例:

    @GetMapping("/example")
    public String exampleMethod(@RequestParam("name") String paramName) {// 方法逻辑return "result";
    }
    

    在这个示例中,@RequestParam("name") 表示要从 HTTP 请求中获取名为 “name” 的查询参数,并将其值绑定到 paramName 这个方法参数上。

  2. @Param:
    @Param 注解通常用于一些 ORM 框架中,如MyBatis,用于指定数据库查询中的参数。它通常用于动态构建SQL查询语句时,指定参数的名称或别名。在MyBatis中,可以通过 @Param 注解给参数起一个名称,以便在XML映射文件中引用。

    示例:

    List<User> getUsersByNameAndAge(@Param("name") String name, @Param("age") int age);
    

    在这个示例中,@Param("name")@Param("age") 分别指定了两个方法参数的名称,在对应的XML映射文件中可以通过这些名称来引用这些参数。

总结来说,@RequestParam 注解是用于后端框架中获取HTTP请求参数的值,而 @Param 注解则是用于ORM框架中指定SQL查询参数的名称。它们在不同的上下文中使用,具有不同的含义和功能。

三层架构

在经典的三层架构中,service层、mapper层和controller层分别表示三个逻辑层次:

  1. service层:业务逻辑层,负责处理具体的业务逻辑,是业务逻辑的实现层。它通常与DAO层(或mapper层)和controller层打交道,接收请求,组织数据,调用DAO层进行数据库操作,最后将结果返回给调用者。

  2. mapper层(或DAO层):数据访问层,负责与数据库进行交互,执行SQL语句,进行数据的读取、写入和修改等操作。这一层把数据库的操作封装成方法,供service层调用,让service层更专注于业务逻辑的处理。

  3. controller层:表示层,负责接收用户的请求,并将请求转发给service层进行处理,最后将处理结果返回给客户端。它通常是Web应用程序中的MVC框架中的控制器部分,负责处理HTTP请求、验证用户输入、调用service层处理业务逻辑等操作。

总之,service层、mapper层和controller层是三个不同的逻辑层,各自负责不同的任务。其中,service层处理具体的业务逻辑,mapper层与数据库进行交互,controller层负责接收请求和返回响应。它们之间通过定义清晰的接口和交互方式来实现松耦合的架构。
举一个简单的例子理解三层架构,以
当前端传过来请求时,controller层也就是表示层直接与前端交互,该层也接收从前端传过来的参数
以对部门操作为例

表示层

package com.example.xh.controller;import com.example.xh.entity.Dept;
import com.example.xh.vo.Result;
import com.example.xh.service.DeptService;
import com.example.xh.vo.UpdateDeptVo;
import jakarta.annotation.Resource;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.*;import java.util.List;
import java.util.Objects;
//要有注解,表示这是一个controller层
@RestController
//该层可能有好几个类,怎么表示应该进入这个类?用路径表示
@RequestMapping("/depts")
public class DeptController {//service层注入controller层,controller层调用service层的方法@Resourceprivate DeptService deptService;//查询 这里通过指明发起请求的方式来判定进入哪个方法@GetMappingpublic Result queryAllDepts(){List<Dept> depts = deptService.queryAllDepts();//直接调用service的方法return Result.success(depts);//controller层返回的是一个结果类//里面包含了请求是否成功,以及如果是请求数据,会把数据传过去}//更新@PutMappingpublic Result updateDept(@RequestBody UpdateDeptVo vo){//@RequestBody注释表示接收来自前端的传参,如果是json传参,需要用@RequestBody表示//检查传参是否正确可以在controller层做if(Objects.isNull(vo.getId())){return  Result.error("id不能为空");}if(StringUtils.isEmpty(vo.getName())){return  Result.error("Name不能为空");}//直接调用service的方法int i = deptService.updateDept(vo.getId(),vo.getName());return i>0?Result.success():Result.error("更新失败");}//插入@PostMappingpublic  Result insertDept(@RequestBody UpdateDeptVo vo){//检查传参是否正确if(StringUtils.isEmpty(vo.getName())){return  Result.error("Name不能为空");}return deptService.insertDept(vo.getName());}//删除@DeleteMapping("/{id}")//当使用url传参中的路径传参时,采用@PathVariable接收参数//@PathVariable("id") int Id表示将前端传过来的id与后端的Id绑定public Result deleteById(@PathVariable("id") int id){int i = deptService.deleteById(id);return i>0?Result.success():Result.error("删除失败");}
}

Result的封装

这里是返回给前端的数据,一般是前后端进行协调,什么样的返回代表什么样的含义

package com.example.xh.vo;import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
//自动产生get和set方法
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Result {private Integer code;//响应码,1 代表成功; 0 代表失败private String msg;  //响应信息 描述字符串private Object 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);}
}

表示层接收完消息后,并进行简单的参数验证后,调用服务层的函数实现对应的功能,那么主要的操作要放在服务层。服务层一般有两种类,接口类和接口的实现类,降低代码的耦合。
接口类

package com.example.xh.service;import com.example.xh.entity.Dept;
import com.example.xh.vo.Result;import java.util.List;public interface DeptService {List<Dept> queryAllDepts();int updateDept(int id,String name);Result insertDept(String name);int deleteById(int id);
}

接口的实现类

package com.example.xh.service.impl;import com.example.xh.entity.Emp;
import com.example.xh.mapper.EmpMapper;
import com.example.xh.service.EmpService;
import com.example.xh.vo.PageVo;
import com.example.xh.vo.Result;
import com.github.pagehelper.Page;
import com.github.pagehelper.PageHelper;
import jakarta.annotation.Resource;
import org.springframework.stereotype.Service;import java.time.LocalDateTime;
import java.util.List;
import java.util.Objects;//业务层
@Service
public class EmpServiceImp implements EmpService {@Resourceprivate EmpMapper empMapper;@Overridepublic Result queryEmpsByPage(String name, Integer gender, String begin, String end, Integer page, Integer pageSize) {//开启分页Page<Object> objects = PageHelper.startPage(page,pageSize);//查询数据List<Emp> emps = empMapper.queryEmpsByPage(name, gender, begin, end, page, pageSize);//查询出的数据放置在pagevo里PageVo<Emp> pageVo = new PageVo<>();pageVo.setRows(emps);pageVo.setTotal(objects.getTotal());//把page对象放在result里return Result.success(pageVo);}@Overridepublic void delete(List<Integer> ids) {empMapper.deteleByIfs(ids);}@Overridepublic Result save(Emp emp) {//检查某些操作是否合法,放在服务层里去做Emp emp1 = empMapper.selectEmpByName(emp.getUsername());if(Objects.nonNull(emp1)){return Result.error("用户名"+emp.getUsername()+"重复");}//需要新添加一些内容,也放在服务层里去做,dao层就是对数据库操作,其它不用管emp.setUpdateTime(LocalDateTime.now());emp.setCreateTime(LocalDateTime.now());int i =  empMapper.insertEmp(emp);return i>0?Result.success():Result.error("添加失败");}@Overridepublic Emp findById(Integer id) {return  empMapper.findById(id);}@Overridepublic Result update(Emp emp) {Emp emp2 = empMapper.findById(emp.getId());//修改的用户名和以前的用户名不相同,就检查新用户名是不是合法的,即不重复if(!emp2.getUsername().equals(emp.getUsername())){Emp emp1 = empMapper.selectEmpByName(emp.getUsername());if(Objects.nonNull(emp1)){return Result.error("用户名"+emp.getUsername()+"重复");}}emp.setUpdateTime(LocalDateTime.now());int i =  empMapper.update(emp);return i>0?Result.success("更新成功"):Result.error("更新失败");}
}

最后就是Dao层,基本是写sql语句

package com.example.xh.mapper;import com.example.xh.entity.Dept;
import org.apache.ibatis.annotations.*;import java.util.List;@Mapper
public interface DeptMapper {//查询@Select("select id,name,update_time from dept")List<Dept> queryAllDepts();//根据姓名查询  用在插入,当插入相同名字部门时显示插入错误@Select("select * from dept where name = #{name}")Dept selectDeptByName(@Param("name") String name);//@Param("name") String Name 将后端的Name与数据库里的name绑定//更新@Update("update dept set name=#{name},update_time=now() where id=#{id}")int updateDept(@Param("id") int id,@Param("name") String name);//插入 测试成功@Insert("insert into dept(name,create_time,update_time) values (#{name},now(),now())")int insertDept(@Param("name") String name);//删除 测试成功@Delete("delete from dept where id = #{id}")int deleteById(@Param("id") int id);
}

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

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

相关文章

【网络安全 | Misc】normal_png

方法一 可以通过stegsolve或winhex看到图片高度被改写&#xff1a; 改为&#xff1a; 再保存图片即可&#xff1a; flag{B8B68DD7007B1E406F3DF624440D31E0}方法二 使用脚本查看宽高是否被修改&#xff1a; import zlib import struct import argparse import itertoolspars…

Java强软弱虚引用

面试&#xff1a; 1.强引用&#xff0c;软引用&#xff0c;弱引用&#xff0c;虚引用分别是什么&#xff1f; 2.软引用和弱引用适用的场景&#xff1f; 3.你知道弱引用的话&#xff0c;能谈谈WeakHashMap吗&#xff1f; 目录 一、Java引用 1、强引用&#xff08;默认支持模式…

【C++】STL 容器 - multiset 容器 ( std::multiset 容器简介 | std::multiset 容器 常用操作 api 简介 )

文章目录 一、mulset 容器1、std::multiset 容器简介2、代码示例 - multiset 容器 二、std::multiset 容器 常用操作 api 简介1、常用 api 简介2、代码示例 - multiset 容器常用操作 一、mulset 容器 1、std::multiset 容器简介 在 C 语言 的 标准模板库 ( STL , Standard Temp…

数据库的学习笔记——第一篇

SQL通用语法 SQL语句 DDL 数据定义 数据库、表字段 DML 数据操作 增删改 DQL 数据查询 查询表中记录 DCL 数据控制 创建用户、控制用户权限 DLL语句——数据库操作 SHOW DATABASES; # 查询数据库SELECT DATABASE(); # 查询当前数据库CREATE DATABASE [IF …

【数据结构】六、树和二叉树

目录 一、树的基本概念 二、二叉树 2.1二叉树的性质 2.2二叉树的存储结构 2.3遍历二叉树 先序遍历 中序遍历 后序遍历 层次遍历 2.4二叉树的应用 计算叶子数 前序遍历建树 根据序列恢复二叉树 计算树的深度 判断完全二叉树 三、线索二叉树 3.1线索化 四、树和森林…

Scala知识点——App类

我们在代码中一般程序都是是通过main方法进入。但是在scala中提供了一个App类&#xff0c;通过继承可以实现不用显式的调用main方法就能运行。 App类中实现了main方法&#xff1a;

迪杰斯特拉(Dijkstra)算法详解

【专栏】数据结构复习之路 这篇文章来自上述专栏中的一篇文章的节选&#xff1a; 【数据结构复习之路】图&#xff08;严蔚敏版&#xff09;两万余字&超详细讲解 想了解更多图论的知识&#xff0c;可以去看看本专栏 Dijkstra 算法讲解&#xff1a; 迪杰斯特拉算法(Di…

前端基础(三十七):属性结构数据进行关键字筛选

效果 核心源码 type MenuItem {label: string;key: string | number;icon?: React.ReactNode;children?: MenuItem[];type?: group; }function filterTreeData(tree: MenuItem[], keyword: string): MenuItem[] {return tree.filter((node: MenuItem) > {if (node.labe…

SUMO Reward Points v29.8.0WooCommerce 奖励系统插件WORDPRESS积分奖励系统

SUMO Reward Points v29.8.0WooCommerce 奖励系统插件WORDPRESS积分奖励系统 SUMO 奖励积分&#xff1a;WooCommerce 的忠诚度解决方案 一、引言 SUMO 奖励积分&#xff0c;v29.8.0&#xff0c;是一个卓越的WooCommerce插件&#xff0c;致力于为电商提供一套完整的忠诚度奖励系…

matplotlib单变量和双变量可视化

使用seaborn 库的tips数据集&#xff0c;其中包含了某餐厅服务员收集的顾客付小费的相关数据&#xff08;评论区&#xff09; 单变量可视化 直方图 直方图是观察单个变量最常用的方法。这些值是经过"装箱"&#xff08;bin&#xff09;处理的 直方图会将数据分组后绘…

IntelliJ IDEA [插件 MybatisX] mapper和xml间跳转

文章目录 1. 安装插件2. 如何使用3. 主要功能总结 MybatisX 是一款为 IntelliJ IDEA 提供支持的 MyBatis 开发插件 它通过提供丰富的功能集&#xff0c;大大简化了 MyBatis XML 文件的编写、映射关系的可视化查看以及 SQL 语句的调试等操作。本文将介绍如何安装、配置和使用 In…

B/S架构云端SaaS服务的医院云HIS系统源码,自主研发,支持电子病历4级

医院云HIS系统源码&#xff0c;自主研发&#xff0c;自主版权&#xff0c;电子病历病历4级 系统概述&#xff1a; 一款满足基层医院各类业务需要的云HIS系统。该系统能帮助基层医院完成日常各类业务&#xff0c;提供病患挂号支持、病患问诊、电子病历、开药发药、会员管理、统…

Spring漏洞合集

目录 什么是spring区分Spring与Struts2框架的几种新方法CVE-2016-4977&#xff1a;Spring Security OAuth2 远程命令执行漏洞漏洞介绍 & 环境准备漏洞发现漏洞验证 & 利用1利用2 CVE-2017-4971&#xff1a;Pivotal Spring Web Flow 远程代码执行漏洞漏洞介绍 & 环境…

使用Pytorch搭建模型

本来是只用Tenorflow的&#xff0c;但是因为TF有些Numpy特性并不支持&#xff0c;比如对数组使用列表进行切片&#xff0c;所以只能转战Pytorch了&#xff08;pytorch是支持的&#xff09;。还好Pytorch比较容易上手&#xff0c;几乎完美复制了Numpy的特性&#xff08;但还有一…

广州市网约车平台:照片上传标准与处理技巧全解析

《广州市网络预约出租汽车综合业务管理平台》是一个旨在规范和提升广州市网约车服务质量的在线管理系统。它整合了司机和车辆信息管理、预约服务、监管监控、服务质量评价以及数据分析等功能&#xff0c;确保网约车服务符合当地法规要求&#xff0c;同时为乘客提供安全、便捷的…

浅谈大模型推理成本优化

上回说了&#xff0c;全赞AI的应用里面有用到几十个大模型&#xff0c;我的其他的应用比如渣渣句&#xff0c;熊喵表情都会或多或少的用到一到两个大模型的推理。而众所周知&#xff0c;目前大模型的推理存在两个问题&#xff0c;一个是慢&#xff0c;一个是贵&#xff0c;慢的…

微服务系列之分布式事务理论

概述 事务是由一组操作构成的可靠的独立的工作单元&#xff0c;事务具备ACID的特性&#xff0c;即原子性、一致性、隔离性和持久性。 分类 大多数情况下&#xff0c;分类是没有意义的一件事。但是分类可以一定程度上&#xff0c;加深理解。 实现 从实现角度来看&#xff0…

轻松删除文件名中的符号,使用替换功能,让管理文件更加得心应手!

在我们的日常生活和工作中&#xff0c;文件管理是一项必不可少的任务。而一个整洁、有序的文件名系统则有助于我们快速找到所需的文件。如果你发现文件名中存在一些不必要的符号&#xff0c;那么这款文件重命名工具将是你的得力助手。它具备强大的替换功能&#xff0c;可以轻松…

在 Unity 中获取 Object 对象的编辑器对象

有这个需求的原因是&#xff0c;在编辑器的 Inspector 逻辑中&#xff0c;写了许多生成逻辑。 现在不想挨个在 Inspector 上都点一遍按钮&#xff0c;所以就需要能获取到它们的编辑器对象。 发现可以借助官方的 UnityEditor.Editor.CreateEditor 方法达到目的&#xff0c;如下…

OpcUaHelper实现西门子OPC Server数据交互

Opc ua客户端类库,基于.net 4.6.1创建,基于官方opc ua基金会跨平台库创建,方便的实现和OPC Server进行数据交互。 FormBrowseServer 在开发客户端之前,需要使用本窗口来进行查看服务器的节点状态,因为在请求服务器的节点数据之前,必须知道节点的名称,而节点的名称可以…