SpringBoot+MyBatis-plus实现CRUD (踩坑总结!!!)

一、创建项目,引入相应的依赖 (项目源码在文末)

(不要选Module !!!!)

 

 <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.4.2</version></dependency>

 二、创建表和对应的实体类

CREATE TABLE `user` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`user_name` varchar(255) DEFAULT NULL COMMENT '用户名',`user_password` varchar(255) DEFAULT NULL COMMENT '密码',`email` varchar(255) DEFAULT NULL COMMENT '邮箱',`phone` varchar(255) DEFAULT NULL COMMENT '手机号',`create_time` datetime DEFAULT NULL COMMENT '创建时间',PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='用户表';
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.Date;@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {@TableId(value = "id", type = IdType.AUTO)private Long id;//用户名private String userName;//密码private String userPassword;//邮箱private String email;//手机号private String phone;//创建时间private Date createTime;
//    private String createTime;
}

 三、创建对应的实现层和控制层

import com.example.registerdemo.entity.User;
import com.example.registerdemo.param.UserRegisterParam;
import com.example.registerdemo.pojo.UserDTO;
import com.example.registerdemo.service.UserService;
import org.springframework.beans.BeanUtils;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.*;
import javax.annotation.Resource;
import javax.validation.constraints.NotNull;
import java.util.List;
@RestController
@RequestMapping("/user")public class UserController {@Resourceprivate UserService userService;//用户注册,用于处理用户注册请求@PostMapping("/register")public String userRegister(@Validated @RequestBody UserRegisterParam param) {User user = new User();BeanUtils.copyProperties(param,user);if(userService.userRegister(user)) {return "注册成功";}return "注册失败";}//获取用户的所有信息,用于处理查询所有用户的信息的请求@PostMapping("/getAllUser")public List<User> getAllUserInfo() {return userService.getAllUserInfo();}@PostMapping("/getUserByName")@Validatedpublic List<User> getUser(@RequestParam("username") @NotNull(message = "用户名不能为空") String name){return userService.getUsersByName(name);}/*** 更新用户信息*/@PostMapping("/getUpdateUser/{id}")public List<User> getUpdateUser(@RequestBody UserDTO userDTO){return userService.updateUser(userDTO);}/*** 删除用户*/@PostMapping("/deleteUser/{id}")public void deleteUser(@PathVariable Long id) {userService.deleteUser(id);}
}
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.example.registerdemo.entity.User;
import com.example.registerdemo.mapper.UserMapper;
import com.example.registerdemo.pojo.UserDTO;
import com.example.registerdemo.service.UserService;
import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Service;
import org.springframework.util.DigestUtils;
import org.springframework.util.StringUtils;import javax.annotation.Resource;
import java.util.Date;
import java.util.List;@Service
public class RegisterUserServiceImpl implements UserService {@Resourceprivate UserMapper userMapper;@Override/*** 用户注册功能(插入)** 通过当前日期时间获取格式为 "yyyy-MM-dd HH:mm:ss" 的字符串 createTime。* 设置用户对象的 createTime 属性为 createTime。* 插入用户对象到数据库表中,如果受影响的行数大于0,则返回 true,否则返回 false。*/public boolean userRegister(User user) {user.setCreateTime(new Date());if(userMapper.insert(user)>0) {return true;}return false;}/*** 查询全部* @return*/@Overridepublic List<User> getAllUserInfo() {return userMapper.selectList(null);}/*** 按照姓名进行查询* @param name* @return*/@Overridepublic List<User> getUsersByName(String name) {QueryWrapper<User> queryWrapper = new QueryWrapper<>();queryWrapper.eq("name", name);return userMapper.selectList(queryWrapper);}/*** 按照id进行更新** @param userDTO* StringUtils.isBlank和StringUtils.isEmpty主要区别就是* StringUtils.isBlank会认为全是空格的字符串为空,而StringUtils.isEmpty则将空格字符串认为不是空*/@Overridepublic List<User> updateUser( UserDTO userDTO) {User user = new User();BeanUtils.copyProperties(userDTO, user);user.setId(userDTO.getId());if (!StringUtils.isEmpty(userDTO.getUserPassword())) {user.setUserPassword(userDTO.getUserPassword());}userMapper.updateById(user);return null;}/*** 按照id进行删除* @param id*/@Overridepublic void deleteUser(Long id) {userMapper.deleteById(id);}
}

四、踩坑总结

        当然我没有单纯的从这个CRUD的内容上讨论踩坑,但是在这个过程中,遇到了一点相关的问题相信后面可能也会遇到,就在此进行一个总结吧。

踩坑①、

        我在写 “用户注册功能(其实就是简单的插入)”的时候,使用了BeanUtils工具类中的copyProperties的方法(当然,这是一种浅拷贝的方法,这只能拷贝同类型的数据,

先来说一下什么是浅拷贝,深拷贝。这两者最主要的区别就在于是否是复制了对象的真实实体还是说只是使用了一个指针,两者指向的是内存中的同一个对象。

浅拷贝(shallowCopy)只是增加了一个指针指向已存在的内存地址;

深拷贝(deepCopy)是增加了一个指针并且申请了一个新的内存,使这个增加的指针指向这个新的内存;同时如果一个类中包含有引用类型变量,则该类拷贝以后引用类型也会被拷贝。)

这里面还有一个易错点就是,谁拷贝谁,不要弄反了!

点进方法的源码进行查看,可以发现:

 

方法中第一个元素的位置是,原始数据,即传进来的数据,第二位置放的数据是目标对象的数据,即复制后的数据,后面要使用的也就是第二个方法的数据,这两个位置的顺序一定不能反。

踩坑②、

        在进行用户注册的时候,还加入了“参数校验”的功能,然而我第一次引用的依赖是这个,导致我参数校验的时候,发现 @Validated注解根本没有起作用。

  <dependency><groupId>javax.validation</groupId><artifactId>validation-api</artifactId><version>1.1.0.Final</version></dependency>

后来修改成下面引用的依赖后,注解正常有效果了 。

 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-validation</artifactId></dependency>

 踩坑③、

        在创建时间属性时,发现生成的时间,总是和真实的时间没有对应上(其实就是时间的时区没有对应上!!),后面在时间的属性上,加入了注解@DateTimeFormat,@JsonFormat 好像也没有生效。

 这个尝试了很多方法也没有效果,后面换了一个思路,尝试在yml文件里面配置看看,然后发现了这篇文章:

 搜索的关键词有的时候也挺重要的

 

spring:application:name: registerdemodatasource:driver-class-name: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://localhost/register?useUnicode=true&characterEncoding=utf-8&useSSL=true&serverTimezone=GMT%2b8&allowMultiQueries=true

这样配置好在运行就好了,那个时间格式化的注解也不用加了。 

最后还要注意的一个小点就是 @Mapper 不要忘记加了。

 

整个demo的源码详见:mybatis-plus_CRUD · 企业级代码管理平台 (aliyun.com)

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

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

相关文章

Linux 常用命令: 查看 CPU 信息

强力推荐-不要错过&#xff0c;万一能帮助到自己呢&#xff1f; 推荐一个网站&#xff0c;关于人工智能教程&#xff0c;教程不仅是零基础&#xff0c;通俗易懂&#xff0c;而且非常风趣幽默&#xff0c;像看小说一样&#xff01;觉得太牛了&#xff0c;所以分享给大家。点 这…

js获取body或屏幕的宽度大集合

Js代码 //【转】js获取body或屏幕的宽度大集合 var s " "; document.documentElement.scrollTop 就是滚动条距离顶部的位置&#xff08;可变&#xff09; document.documentElement.scrollLef 指滚动条距离左边的位置&#xff08;可变&#xff0…

js获取屏幕、浏览器、页面的宽度和高度

介绍 容器的尺寸是指当前分辨率下的高度和宽度&#xff08;分辨率不同&#xff0c;值就不相同&#xff09; 屏幕信息 screen.height&#xff1a;屏幕高度 screen.availHeight&#xff1a;屏幕可用宽度 screen.height - acreen.availHeight&#xff1a;任务栏高度 浏览器信息…

PHPMySQL基础(四):模拟登录Login功能案例

PHP&MySQL基础&#xff08;一&#xff09;:创建数据库并通过PHP进行连接_长风沛雨的博客-CSDN博客 PHP&MySQL基础&#xff08;二&#xff09;:通过PHP对MySQL进行增、删、改、查_长风沛雨的博客-CSDN博客 PHP&MySQL基础&#xff08;三&#xff09;:处理查询SQL返…

《物联网IoT解决方案》(Unity+SteamVR+云技术+5G+AI+物联网+IoT+人机交互+万物互联+物物互联+射频识别+全球定位系统+实时采集+智能化感知+识别+管理+立钻哥哥+==)

《物联网IoT解决方案》 版本 作者 参与者 完成日期 备注 YanlzVR_IoT_V01_1.0 严立钻 2020.05.05 ##《物联网IoT解决方案》发布说明&#xff1a; “物联网IoT解决方案”&#xff1a;是对“IoT”的基础探索&#xff1b;【VR云游戏】&#xff1a;U…

创建联系人vcf文件通讯录

现在的app动不动就要访问手机联系人&#xff0c;于是突发奇想把自己手机里面导很多假联系人&#xff0c;让他去获取。于是随便写了一下代码生成了.vcf后缀的联系人文件&#xff0c;可以直接导入到手机里面去。下面代码一些写的不规范&#xff1a;比如流的关闭、异常的处理、数据…

CAD测面积周长

点取闭合区域中的一点&#xff0c;计算闭合区域的面积。 1.单击ET”工具栏的“面积周长命令”按钮。 2.在命令行中输入ET_CaleArea&#xff0c;按回车键。 执行命令后&#xff0c;命令行将显示如下信息&#xff1a; 命令: MxET_CaleArea 拾取闭合区域内部点: 计算的面积为…

怎样去测量CAD面积?

2019独角兽企业重金招聘Python工程师标准>>> 我们在利用CAD看图软件查看CAD图纸时&#xff0c;有时候需要对里面的CAD图形面积进行测量&#xff0c;那么我们应该怎么测量CAD图纸面积&#xff1f;怎样用CAD测量面积&#xff1f;CAD面积怎么算&#xff1f; 1.准备一款…

ROS:发布者Publisher的编程实现(C++)

目录 一、话题模型二、创建功能包三、创建Publisher代码四、编译代码五、运行 一、话题模型 图中&#xff0c;我们使用ROS Master管理节点。 有两个主要节点&#xff1a; Publisher&#xff0c;名为Turtle Velocity&#xff08;即海龟的速度&#xff09; Subscriber&#xff0c…

GlobalMapper20:10分钟根据CAD、shp等离散高程点生成一份精准边界的地形数据

序&#xff1a; 做工程设计&#xff0c;规划设计也好&#xff0c;经常性需要地形数据&#xff0c;而到手的数据多数为excel、txt等文本文件&#xff0c;或者等高线等cad文件。 设计方案汇报是设计项目经理的主要职责&#xff0c;经常需要把设计中的地形&#xff08;shp、cad、…

CAD中怎么批量标注坐标?CAD批量标注坐标​

在CAD绘图过程中&#xff0c;当需要进行有大量CAD坐标标注时&#xff0c;该如何操作呢&#xff1f;下面就来和小编一起来了解一下浩辰CAD建筑中关于批量标注普通坐标的相关操作步骤吧&#xff01; CAD批量标注普通坐标操作步骤&#xff1a; 首先打开浩辰CAD建筑软件&#xff…

ARCGIS格式转CAD时保留其图层、高程及颜色的方法

ARCGIS格式转CAD时保留其图层、高程及颜色的方法 一.转换时&#xff0c;保留高程值 通常情况&#xff0c;shp转CAD格式过程中会丢失某些需要的字段&#xff08;如高程&#xff09;&#xff0c;这是因为CAD无法识别这些字段。如果需要保留某字段的信息&#xff0c;转换之前需添…

CAD导入Revit缺少东西原因-Revit中如何批量导出CAD图纸

一、CAD导入Revit缺少东西原因汇总 在Revit中导入CAD进行模型搭建是建模过程中常用的方法&#xff0c;但是有时会遇到导入的CAD缺少东西的情况&#xff0c;下面介绍几种导致这种问题的原因 1.CAD导入的时候&#xff0c;不是设置为全部可见。 CAD导入Revit中时&#xff0c;“图层…

CAD图纸导入REVIT内并精准建模

整体原理逻辑&#xff1a; 1、CAD图纸(原点至原点&#xff09;导入&#xff0c;作为建模参考底图&#xff0c;此刻测量点、项目基点位置都与CAD图纸原点重合。 2、测量点位置不动&#xff08;目的是为了保障明确参考坐标系原点的坐标&#xff09;&#xff0c;挪动项目基点位置…

CAD中怎么绘制攒尖屋顶?CAD设计攒尖屋顶技巧

在给排水CAD设计中&#xff0c;有些时候为了需要会在图纸中绘制攒尖屋顶&#xff0c;那么你知道CAD软件中怎么构造攒尖屋顶三维模型吗&#xff1f;其实很简单&#xff0c;浩辰CAD给排水软件中提供了实用的攒尖屋顶功能&#xff0c;下面就和小编一起来看看浩辰CAD给排水软件中CA…

CAD中如何识别CAD标高范围?

说到CAD标高&#xff0c;相信很多CAD制图初学入门小伙伴都有所了解&#xff0c;那么浩辰CAD给排水软件中如何识别标高范围呢&#xff1f;接下来的CAD制图初学入门教程就让小编来给大家介绍一下正版CAD软件——浩辰CAD给排水软件中识别CAD标高范围的相关操作技巧吧&#xff0c;感…

CAD地形图等高线标高批量取整工具,解决等高线标高出现小数的问题,等高线高程批量取整,在指定限差内将等高线标高修改为最接近的整数

目录 一、实现效果 二、实现过程 1.暴露等高线标高 2.获取标高小数 3.标高取整 4.标高重新赋值及数据分类输出 5.工具封装 地形图等高线要求高程值须为整数&#xff0c;而在实际生产中&#xff0c;因为某些原因&#xff0c;会出现部分等高线的标高不为整数的情况&#x…

Revit中导入的CAD标高不统一处理及“标高管理”

一、Revit中如何处理导入的CAD标高不统一情况 在Revit中CAD时&#xff0c;尤其是在导入地形CAD文件时&#xff0c;有时会遇到导不进去的情况&#xff0c;这有可能是导入的CAD文件某一个图层标高过大&#xff0c;超出一定范围的原因&#xff0c;如图1所示。 对于这种问题&#x…

CAD中怎么识别标高范围?CAD标高范围识别技巧

说到CAD标高&#xff0c;相信很多CAD制图初学入门小伙伴都有所了解&#xff0c;那么浩辰CAD给排水软件中如何识别标高范围呢&#xff1f;接下来的CAD制图初学入门教程就让小编来给大家介绍一下正版CAD软件——浩辰CAD给排水软件中识别CAD标高范围的相关操作技巧吧&#xff0c;感…

CAD中怎么标注井底标高?CAD井底标高教程

在绘制CAD图纸的过程中&#xff0c;经常会需要添加各种CAD标注&#xff0c;比如&#xff1a;尺寸标注&#xff0c;角度标注等。但是井标注可能很多小伙伴没有听过&#xff0c;接下来的CAD教程就让小编来给大家介绍一下浩辰CAD给排水软件的专业CAD标注中井标注的相关使用技巧吧&…