Spring快速整合Mybatis

MyBatis是一个优秀的持久层框架,Spring则是广泛使用的Java应用框架。可以将两者整合可以充分发挥各自的优势。

1、Spring整合MyBatis的基本配置

添加依赖:

<dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.3.10</version>
</dependency>
<!-- Spring JDBC支持 -->
<dependency><groupId>org.springframework</groupId><artifactId>spring-jdbc</artifactId><version>5.3.10</version>
</dependency>
<!-- MyBatis核心 -->
<dependency><groupId>org.mybatis</groupId><artifactId>mybatis</artifactId><version>3.5.7</version>
</dependency>
<!-- MyBatis-Spring整合包 -->
<dependency><groupId>org.mybatis</groupId><artifactId>mybatis-spring</artifactId><version>2.0.6</version>
</dependency>
<!-- 数据库驱动 -->
<dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>8.0.26</version>
</dependency>

Spring配置文件:
创建Spring配置文件applicationContext.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttps://www.springframework.org/schema/context/spring-context.xsd"><!-- 配置数据源 --><bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"><property name="driverClassName" value="com.mysql.cj.jdbc.Driver"/><property name="url" value="jdbc:mysql://localhost:3306/mybatis_db?useSSL=false&amp;serverTimezone=UTC"/><property name="username" value="root"/><property name="password" value="password"/></bean><!-- 配置SqlSessionFactory --><bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"><property name="dataSource" ref="dataSource"/><!-- 指定MyBatis配置文件 --><property name="configLocation" value="classpath:mybatis-config.xml"/><!-- 指定mapper.xml文件位置 --><property name="mapperLocations" value="classpath:mapper/*.xml"/></bean><!-- 配置Mapper扫描器 --><bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"><property name="basePackage" value="com.example.mapper"/><property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/></bean><!-- 开启注解扫描 --><context:component-scan base-package="com.example"/>
</beans>

MyBatis配置文件:
创建mybatis-config.xml:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configurationPUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration><settings><setting name="mapUnderscoreToCamelCase" value="true"/></settings><typeAliases><package name="com.example.entity"/></typeAliases>
</configuration>

2、基本的CRUD操作

创建实体类:

package com.example.entity;
public class User {private Long id;private String username;private String password;private String email;// 省略getter和setter方法// 省略toString方法
}

创建Mapper接口:

package com.example.mapper;
import com.example.entity.User;
import java.util.List;
public interface UserMapper {int insert(User user);int update(User user);int deleteById(Long id);User selectById(Long id);List<User> selectAll();
}

创建Mapper XML文件:
在resources/mapper目录下创建UserMapper.xml:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.mapper.UserMapper"><insert id="insert" parameterType="User" useGeneratedKeys="true" keyProperty="id">INSERT INTO user(username, password, email)VALUES(#{username}, #{password}, #{email})</insert><update id="update" parameterType="User">UPDATE userSET username=#{username}, password=#{password}, email=#{email}WHERE id=#{id}</update><delete id="deleteById" parameterType="long">DELETE FROM user WHERE id=#{id}</delete><select id="selectById" parameterType="long" resultType="User">SELECT * FROM user WHERE id=#{id}</select><select id="selectAll" resultType="User">SELECT * FROM user</select>
</mapper>

创建Service层:

package com.example.service;
import com.example.entity.User;
import com.example.mapper.UserMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
@Service
public class UserService {@Autowiredprivate UserMapper userMapper;public int addUser(User user) {return userMapper.insert(user);}public int modifyUser(User user) {return userMapper.update(user);}public int removeUser(Long id) {return userMapper.deleteById(id);}public User getUserById(Long id) {return userMapper.selectById(id);}public List<User> getAllUsers() {return userMapper.selectAll();}
}

测试类:

import com.example.entity.User;
import com.example.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import java.util.List;
public class Application {public static void main(String[] args) {ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");UserService userService = context.getBean(UserService.class);// 添加用户User user = new User();user.setUsername("testUser");user.setPassword("123456");user.setEmail("test@example.com");userService.addUser(user);System.out.println("添加用户成功,ID: " + user.getId());// 查询用户User dbUser = userService.getUserById(user.getId());System.out.println("查询用户: " + dbUser);// 更新用户dbUser.setEmail("newemail@example.com");userService.modifyUser(dbUser);System.out.println("更新用户成功");// 查询所有用户List<User> users = userService.getAllUsers();System.out.println("所有用户:");users.forEach(System.out::println);// 删除用户userService.removeUser(user.getId());System.out.println("删除用户成功");}
}

3、动态SQL和分页查询

扩展Mapper接口:
// 在UserMapper接口中添加

List<User> selectByCondition(@Param("username") String username, @Param("email") String email);List<User> selectByPage(@Param("offset") int offset, @Param("pageSize") int pageSize);

扩展Mapper XML:

<!--UserMapper.xml中添加 -->
<select id="selectByCondition" resultType="User">SELECT * FROM user<where><if test="username != null and username != ''">AND username LIKE CONCAT('%', #{username}, '%')</if><if test="email != null and email != ''">AND email LIKE CONCAT('%', #{email}, '%')</if></where>
</select>
<select id="selectByPage" resultType="User">SELECT * FROM user LIMIT #{offset}, #{pageSize}
</select>

扩展Service:
// 在UserService中添加

public List<User> getUsersByCondition(String username, String email) {return userMapper.selectByCondition(username, email);
}
public List<User> getUsersByPage(int pageNum, int pageSize) {int offset = (pageNum - 1) * pageSize;return userMapper.selectByPage(offset, pageSize);
}

测试动态SQL和分页:
// 在Application的main方法中添加测试代码

System.out.println("条件查询:");
List<User> conditionUsers = userService.getUsersByCondition("test", null);
conditionUsers.forEach(System.out::println);
System.out.println("分页查询(第1页,每页2条):");
List<User> pageUsers = userService.getUsersByPage(1, 2);
pageUsers.forEach(System.out::println);

4、事务管理

修改Spring配置

在applicationContext.xml中添加事务配置:
<!-- 事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="dataSource"/>
</bean>
<!-- 开启注解驱动的事务管理 -->
<tx:annotation-driven transaction-manager="transactionManager"/>

创建事务性Service方法:

@Transactional
public void transferEmail(Long fromId, Long toId, String newEmail) {User fromUser = userMapper.selectById(fromId);User toUser = userMapper.selectById(toId);if(fromUser == null || toUser == null) {throw new RuntimeException("用户不存在");}// 将fromUser的email设置为newEmailfromUser.setEmail(newEmail);userMapper.update(fromUser);// 模拟异常if(newEmail.contains("error")) {throw new RuntimeException("测试事务回滚");}// 将toUser的email也设置为newEmailtoUser.setEmail(newEmail);userMapper.update(toUser);
}

测试事务:

try {userService.transferEmail(1L, 2L, "transaction@example.com");System.out.println("事务操作成功");
} catch (Exception e) {System.out.println("事务操作失败: " + e.getMessage());
}
try {userService.transferEmail(1L, 2L, "error@example.com");
} catch (Exception e) {System.out.println("事务回滚测试: " + e.getMessage());
}

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

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

相关文章

基于深度学习的语音识别:从音频信号到文本转录

前言 语音识别&#xff08;Automatic Speech Recognition, ASR&#xff09;是人工智能领域中一个极具挑战性和应用前景的研究方向。它通过将语音信号转换为文本&#xff0c;为人们提供了更加自然和便捷的人机交互方式。近年来&#xff0c;深度学习技术在语音识别领域取得了显著…

本地部署Nacos开源服务平台,并简单操作实现外部访问,Windows 版本

Nacos 是一款阿里开源的动态服务发现、配置、管理平台&#xff0c;拥有易于集成、高可用与可扩展等特点。它提供了动态服务注册和发现能力&#xff0c;使得服务自动注册到服务器并且消费真能够发现提供者。本文将详细介绍如何在本地安装 Nacos &#xff0c;以及结合nat123端口映…

数据结构:反转字符串(Reversing a String)

目录 方法一&#xff1a;双指针法 方法二&#xff1a;辅助数组 方法对比总结&#xff1a; 问题定义 给定一个字符串&#xff0c;例如&#xff1a; char str[] "hello";我们的目标是把它反转成&#xff1a; "olleh"&#x1f4cc; 输入特点&#xff…

Redis Copy-on-Write机制:

Copy-on-Write机制&#xff1a; 父子进程共享内存页 当父进程修改数据时&#xff0c;内核会复制被修改的页 这可能导致内存使用量暂时增加 通俗的话描述一下可以用一个生活中的例子来通俗解释 Copy-on-Write&#xff08;写时复制&#xff09; 机制&#xff1a;&#x1f4d6; 比…

iOS加固工具有哪些?从零源码到深度混淆的全景解读

在iOS安全加固领域&#xff0c;不同项目类型对保护需求有着本质差异&#xff1a;“我有源码”与“我只有IPA”两条路径决定了你该用什么工具。本文将从 无需源码处理整个IPA包 到 源码级编译期混淆&#xff0c;分层探讨主流工具如何发挥价值&#xff0c;并附上适配方案建议。工…

Composer 可以通过指定 PHP 版本运行

是的&#xff0c;Composer 可以通过指定 PHP 版本运行&#xff0c;尤其是在服务器上有多个 PHP 版本时&#xff08;如 PHP 7.x 和 PHP 8.x&#xff09;。以下是几种常见方法&#xff1a;方法 1&#xff1a;使用 php 命令指定版本 Composer 依赖系统中的 php 命令&#xff0c;因…

vscode文件颜色,只显示自己更改的文件颜色

这个主要是因为你github git下来以后&#xff0c;用vscode打开会默认显示更改了&#xff0c;你只要在这里先手动取消更改就行了&#xff0c;注意不要把你自己更改的取消了

记录我coding印象比较深刻的BUG

4778&#xff1a;我的BUG噩梦问题描述&#xff1a;DAB播放中关ACC掉电后开ACC&#xff0c;手动切到FM/AM(有时第一次切换出现问题/有时第二次切换出现问题)&#xff0c;FM/AM不记忆关ACC前电台或者FM/AM关ACC掉电后开ACC&#xff0c;手动切到DAB再回到FM/AM&#xff0c;FM/AM不…

Kubernetes集群中Istio mTLS握手失败问题排查与解决方案

Kubernetes集群中Istio mTLS握手失败问题排查与解决方案 在微服务架构中&#xff0c;Istio 提供了基于 Envoy 的服务网格能力&#xff0c;其中 mTLS&#xff08;双向 TLS&#xff09;是确保服务间通信安全的重要机制。但在生产环境中&#xff0c;开发者常常会遇到 mTLS 握手失败…

antd+react+可输入的下拉选择组件

该组件是一个可输入的下拉选择组件&#xff0c;支持从预设选项中选择或手动输入自定义值。组件基于 React 和 Ant Design 实现&#xff0c;具有良好的交互体验和灵活的配置选项。 &#x1f9e0; 核心逻辑分析 1. 状态管理 const [isInput, setIsInput] useState(false); con…

React 面试题库

openAI React 面试题库 以下题库按模块分类&#xff08;React 架构与运行机制、核心 API、Diff 算法与事件机制、Fiber 架构与调度、并发模式与过渡、生命周期及新版生命周期对照、综合源码题、扩展专题、React 与 Vue 对比&#xff09;&#xff0c;并按难度&#xff08;初级…

查看两个tv and 手机模拟器的ip

要查看 Android 模拟器 的 IP 地址&#xff0c;你可以使用 ADB shell 命令来获取。下面是详细步骤&#xff1a;步骤 1&#xff1a;查看已连接的模拟器首先&#xff0c;确保你连接的模拟器已经启动并且连接到 ADB。你可以运行以下命令来查看已连接的设备&#xff1a;adb devices…

从零到一:用C语言构建贪吃蛇(一)- 基础框架与数据结构

资料合集下载链接: ​​https://pan.quark.cn/s/472bbdfcd014​ 第一步:绘制游戏世界 - 定义地图边界 任何游戏都需要一个舞台。在贪吃蛇中,这个舞台就是一个有明确边界的矩形地图。 1. 确定尺寸 根据笔记,我们首先要确定地图的尺寸。使用宏定义(​​#define​​)是…

AWS RDS 排查性能问题

AWS RDS 排查数据库问题 1.查看当前横在执行的SQL select id,user,time,left(info,100) from information_schema.processlist where time>0 and info is not null order by time desc ;2.AWS RDS 查看性能详情查看 Top SQL&#xff0c;AAS最高的几个sql&#xff0c;然后看这…

Baumer工业相机堡盟工业相机如何通过YoloV8深度学习模型实现持械检测(C#代码,UI界面版)

Baumer工业相机堡盟工业相机如何通过YoloV8深度学习模型实现持械检测&#xff08;C#代码&#xff0c;UI界面版&#xff09;工业相机使用YoloV8模型实现持械检测工业相机通过YoloV8模型实现持械检测的技术背景在相机SDK中获取图像转换图像的代码分析工业相机图像转换Bitmap图像格…

在 WPF 启动界面中心加载 GIF 动图

在 WPF 启动界面中心加载 GIF 动图 在 WPF 启动界面中心加载 GIF 动图可以通过多种方式实现。下面我将提供一个完整的解决方案&#xff0c;包括使用第三方库和纯 WPF 实现两种方法。 方法一&#xff1a;使用 WpfAnimatedGif 库&#xff08;推荐&#xff09; 这是最简单可靠的方…

Vue前端路由从入门到精通

目录 第1章:路由的本质与Vue Router的魅力 1.1 什么是前端路由? 1.2 为什么选择Vue Router? 1.3 快速上手:安装与基本配置 1.4 一个小实践:动态欢迎页 第2章:路由配置的进阶玩法 2.1 命名路由:给路由取个名字 2.2 动态路由的深度挖掘 2.3 嵌套路由:页面中的页面…

【Python】SQLAlchemy实现upsert

文章目录✅ 通用思路1. 使用 merge() 方法&#xff08;适用于简单场景&#xff09;2. 使用数据库特定的 UPSERT 功能&#xff08;推荐用于性能和并发安全&#xff09;&#x1f7e2; PostgreSQL: 使用 on_conflict_do_update&#x1f7e1; MySQL: 使用 ON DUPLICATE KEY UPDATE&…

快速入门SwiftUI

SwiftUI的入门难度稍微有点高&#xff0c;但对于比较熟悉Swift的UIKit老手来说阵痛期大概1周以内&#xff0c;两周内能达到UIkit的开发效率&#xff0c;个人总结快速入门路径如下&#xff1a; 第一步 周期&#xff1a;1天 操作&#xff1a;阅读苹果官方demo 目的&#xff1a;…

【n8n教程笔记——工作流Workflow】文本课程(第一阶段)——1、导航编辑器界面(Navigating the editor UI)介绍

https://docs.n8n.io/courses/ 文章目录Navigating the Editor UIGetting startedEditor UI settingsLeft-side panelTop barCanvasNodesFinding nodesAdding nodesNode buttonsSummaryNavigating the Editor UI In this lesson you will learn how to navigate the Editor UI…