线程池操作数据库存在线程安全问题

news/2024/9/11 4:27:07/ 标签: 数据库

目录

1、前言

2、问题

3、解决方法

3.1、方法一:数据库约束

3.2、方法二:使用锁进行线程的约束

4、总结


1、前言

        当前需求为:处理数据,将数据存储到数据库中,在存储的过程中,会先查询该数据是否已经存储在数据库中,由于处理数据过大,需要采用线程池的方式进行同时处理。

2、问题

        采用以上手段编写逻辑时,会存在线程安全问题:边查边插(数据库查询数据不一致问题),导致数据的重复插入。因此需要采用手段约束数据重复的问题。

3、解决方法

3.1、方法一:数据库约束

        给数据库中某个字段设计为唯一索引,即使在查询的时候没有查到相应的数据记录(实际已经存在数据库中),在插入的时候会报错字段重复的错误异常,导致数据记录插入失败,从而达到去重的问题。

3.2、方法二:使用锁进行线程的约束

        根据实际业务场景选择合适的锁策略。乐观锁适用于写入冲突较少的场景,悲观锁适用于写入冲突较多的场景。

        线程同步:使用同步机制,如ReentrantLocksynchronized等,确保同一时间只有一个线程能进行查询和插入操作。下面就使用线程同步机制,代码展示设计:

        mybatis接口代码:

import org.apache.ibatis.annotations.*;public interface UserMapper {@Select("SELECT COUNT(*) FROM user WHERE username = #{username}")int countByUsername(@Param("username") String username);@Insert("INSERT INTO user (username, password, email) VALUES (#{username}, #{password}, #{email})")@Options(useGeneratedKeys = true, keyProperty = "id")int insertUser(User user);
}

        创建服务类代码:

import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class UserService {private final ExecutorService executorService;private final SqlSessionFactory sqlSessionFactory;public UserService(SqlSessionFactory sqlSessionFactory) {this.sqlSessionFactory = sqlSessionFactory;// 创建固定大小的线程池this.executorService = Executors.newFixedThreadPool(10);}public void addUser(User user) {executorService.submit(() -> {try (SqlSession session = sqlSessionFactory.openSession()) {UserMapper userMapper = session.getMapper(UserMapper.class);synchronized (UserService.class) {// 在同步块内执行查询和插入操作if (userMapper.countByUsername(user.getUsername()) == 0) {userMapper.insertUser(user);session.commit(); // 提交事务}}}});}
}

以上只是一个小案例,实际使用场景要按照你的需求来。解释下以上代码:使用了UserService.class作为锁对象,以确保同一时间只有一个线程能够执行查询和插入操作。这样可以避免并发问题,但可能会导致性能瓶颈,特别是在高并发场景下。可以考虑使用更细粒度的锁或者数据库级别的锁(如悲观锁)来优化性能。

4、总结

        根据需求设计线程池操作数据库的逻辑,但是整个过程存在线程安全问题,因此,简单介绍了两个解决方式:①数据库表设置唯一索引字段 ②采用加锁的方式保证线程同步。使用时根据实际场景进行调整,过程写的比较宽泛,只介绍了两种方式,如果想要深入了解,还需进一步学习。当然,我希望可以通过我的小提示给困惑的你一个小小的答案。


学习之所以会想睡觉,是因为那是梦开始的地方。
ଘ(੭ˊᵕˋ)੭ (开心) ଘ(੭ˊᵕˋ)੭ (开心)ଘ(੭ˊᵕˋ)੭ (开心)ଘ(੭ˊᵕˋ)੭ (开心)ଘ(੭ˊᵕˋ)੭ (开心)
                                                                                                        ------不写代码不会凸的小刘


http://www.ppmy.cn/news/1474495.html

相关文章

FPGA(1)--什么是布局与布线

布局与布线是FPGA设计流程中非常关键的步骤,它们的目的是将经过综合的逻辑网表映射到FPGA芯片的物理资源上,并通过电气连接来实现设计的功能。具体来说,布局与布线包括以下工作: 布局(Placement)&#xff1…

9.Python学习:Socket

1.网络通信要素(IP端口传输协议) 2.Socket编程 2.1TCP、UDP协议了解 2.2 Socket流程 服务端有两个socket对象,客户端有一个 3.Socket实战 服务端代码: import socket #创建Socket对象 sksocket.socket() #绑定ip与端口号-使…

BUG解决:postman可以请求成功,但Python requests请求报403

目录 问题背景 问题定位 问题解决 问题背景 使用Python的requests库对接物联数据的接口之前一直正常运行,昨天突然请求不通了,通过进一步验证发现凡是使用代码调用接口就不通,而使用postman就能调通,请求参数啥的都没变。 接口…

day30--56. 合并区间+ 738.单调递增的数字

一、56. 合并区间 题目链接:https://leetcode.cn/problems/merge-intervals/ 文章讲解:https://programmercarl.com/0056.%E5%90%88%E5%B9%B6%E5%8C%BA%E9%97%B4.html 视频讲解:https://www.bilibili.com/video/BV1wx4y157nD 1.1 初见思路 …

防火墙第一次综合实验

DMZ区内的服务器,办公区仅能在办公时间内(9:00-18:00)可以访问,生产区的设备全天可以访问。 办公区设备10.8.2.1不允许访问DMZ区的FTP服务器和HTTP服务器,仅能ping通10.0.3.10 1.先建立拒绝BG到DMZ区的安全策略 2.建立BG到DMZ区的icmp允许 3…

AI推介-大语言模型LLMs之RAG(检索增强生成)论文速览(arXiv方向):2024.06.01-2024.06.20

文章目录~ 1.StackRAG Agent: Improving Developer Answers with Retrieval-Augmented Generation2.FoRAG: Factuality-optimized Retrieval Augmented Generation for Web-enhanced Long-form Question Answering3.Model Internals-based Answer Attribution for T…

springboot仓库管理系统+lw+源码+讲解+调试

第3章 系统分析 在进行系统分析之前,需要从网络上或者是图书馆的开发类书籍中收集大量的资料,因为这个环节也是帮助即将开发的程序软件制定一套最优的方案,一旦确定了程序软件需要具备的功能,就意味着接下来的工作和任务都是围绕…

vue项目实现路由按需加载(路由懒加载)的三种方式

使用异步组件 在使用vue-router配置路由时,可以使用异步组件来实现路由的按需加载。异步组件会在路由被访问时才进行加载,从而实现按需加载的效果。需要注意的是,使用异步组件需要借助webpack的动态import语法来实现。例如: cons…

java 公共字段填充

公共字段填充 1、mybatis-plus2、mybatis 使用注解加aop2.1 自定义注解2.2 自定义切面类2.3 在mapper上添加上自定义的注解 1、mybatis-plus 通过在类上使用如下的注解 TableField(fill FieldFill.INSERT) 是 MyBatis-Plus 中的注解,用于自动填充字段的值。MyBat…

动态规划算法-以中学排课管理系统为例

1.动态规划算法介绍 1.算法思路 动态规划算法通常用于求解具有某种最优性质的问题。在这类问题中,可能会有许多可行解。每一个解都对应于一个值,我们希望找到具有最优值的解。动态规划算法与分治法类似,其基本思想也是将待求解问题分解成若…

算法力扣刷题记录 四十【226.翻转二叉树】

前言 继续二叉树其余操作: 记录 四十【226.翻转二叉树】 一、题目阅读 给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。 示例 1: 输入:root [4,2,7,1,3,6,9] 输出:[4,7,2,9,6,3,1]示例…

three完全开源扩展案例02-跳动的音乐

更多案例尽在https://threelab.cn/ 演示地址 import * as THREE from "three"; import { OrbitControls } from "three/examples/jsm/controls/OrbitControls.js";let mediaElement; let analyser; let scene; let camera; let renderer; let controls; …

DOM(文档对象模型)生命周期事件

前言 DOM 生命周期事件涉及到从创建、更新到销毁 DOM 元素的不同阶段。 ● 我们来看下当HTML文档加载完再执行JavaScript代码 document.addEventListener(DOMContentLoaded, function (e) {console.log(HTML parsed adn DOM tree built!, e); })● 除此之外,浏览…

Electron 简单搭建项目

准备工作 全局安装 node npm创建文件夹,并执行 npm init安装 electron npm i electron --save-dev在 package.json 配置文件中的scripts字段下增加一条start命令: {"scripts": {"start": "electron ."} }由于配置中的入…

Day05-组织架构-角色管理

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 1.组织架构-编辑部门-弹出层获取数据2.组织架构-编辑部门-编辑表单校验3.组织架构-编辑部门-确认取消4.组织架构-删除部门5.角色管理-搭建页面结构6.角色管理-获取数…

STM32智能机器人导航系统教程

目录 引言环境准备智能机器人导航系统基础代码实现:实现智能机器人导航系统 4.1 数据采集模块 4.2 数据处理与导航算法 4.3 通信与网络系统实现 4.4 用户界面与数据可视化应用场景:机器人导航应用与优化问题解决方案与优化收尾与总结 1. 引言 智能机器…

【LeetCode】面试题 16.21. 交换和

质量还不错的一道题,适合用于考察二分法。 1. 题目 2. 分析 求出两个数组的总和,我们令总和少的为less,总和多的为more;如果两个数组的总和是奇数,那么怎么都配不平,直接返回false;如果两个数…

冒泡排序与其C语言通用连续类型排序代码

冒泡排序与其C语言通用连续类型排序代码 冒泡排序冒泡排序为交换排序的一种:动图展示:冒泡排序的特性总结:冒泡排序排整型数据参考代码(VS2022C语言环境): 冒泡排序C语言通用连续类型排序代码对比较的方式更…

uniapp自动升级

一、创建云服务空间(https://unicloud.dcloud.net.cn) 云空间用于关联需要版本控制升级的项目,如果已拥有云空间则省略此步骤。 二、搭建 uni升级中心 - 后台管理系统(升级中心 uni-upgrade-center - Admin) uni-adm…

C++:组合和继承的区别

组合介绍以及与继承对比 什么是组合 (1)composition&#xff0c;组合&#xff0c;就是在一个class内使用其他多个class的对象作为成员 (2)用class tree做案例讲解 (3)组合也是一种代码复用方法&#xff0c;本质也是结构体包含 #include <iostream> #include <vector…