数据库的事务的基本特性,事务的隔离级别,事务隔离级别如何在java代码中使用,使用MySQL数据库演示不同隔离级别下的并发问题
文章目录
- 数据库的事务的基本特性
- 事务的四大特性(ACID)
- 4.1、原子性(Atomicity)
- 4.2、一致性(Consistency)
- 4.3、隔离性(Isolation)
- 4.4、持久性(Durability)
- 事务的隔离级别
- 5.1、事务不考虑隔离性可能会引发的问题
- 1、脏读
- 2、不可重复读
- 3、虚读(幻读)
- 5.2、事务隔离性的设置语句
- 事务隔离级别如何在java代码中使用
- Java代码演示及隔离级别的设置
- 5.3、使用MySQL数据库演示不同隔离级别下的并发问题
- 1、当把事务的隔离级别设置为read uncommitted时,会引发脏读、不可重复读和虚读
- 2、当把事务的隔离级别设置为read committed时,会引发不可重复读和虚读,但避免了脏读
- 3、当把事务的隔离级别设置为repeatable read(mysql默认级别)时,会引发虚读,但避免了脏读、不可重复读
- 4、当把事务的隔离级别设置为Serializable时,会避免所有问题
在上一篇文章中,我们提到了MySQL数据库的事务,那么今天就来具体的谈一谈数据库事务的原理以及在java开发中如何使用数据库的事务的,以及在什么场景中应用不同的事务和事务的隔离级别,以及数据库隔离级别的使用。
数据库的事务的基本特性
事务是并发控制的基本单位,保证事务ACID的特性是事务处理的重要任务,而并发操作有可能会破坏其ACID特性。
所以事务是针对并发而言的,即 对 数据 在并发操作时保驾护航。
**原子性:Atomicity **
**原子性:**在我理解看来是,事务中各项操作,要么全部成功要么全部失败。很有江湖义气一说,同生共死。
一致性:Consistency
**一致性:**我理解的是更侧重结果,事务结束后系统状态是一致的。
隔离性:Isolation
隔离性:并发执行的事务彼此无法看到对方的中间状态。
持久性 :Durability
持久性:当事务完成后,它对于数据的改变是永久性的,即使出现致命的系统故障也将一直保持。
在实际生产应用中 针对 事务的隔离性 又划分出了几种隔离级别

并发事务处理带来的问题
- 更新丢失
当两个或多个事务选择同一行,然后基于最初选定的值更新该行时,由于每个事务都不知道其他事务的存在,就会发生丢失更新问题–最后的更新覆盖了由其他事务所做的更新
- 脏读:在一个事务处理过程中读取了另一个未提交事务中的数据。

解读:两个事务 A 和 B,首先 A 事务对 数据 a 执行加 500 的操作 a = 1500,此时 B 事务读取数据 a 的值 1500,后 A 事务 又对数据 a 执行减500 的操作 a = 1000 ,A 事务 commit 。
不可重复读:事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据时,结果 不一致。

解读:两个事务 A 和 B,首先 A 事务对 数据 a 进行查询 a = 1000,此时 B 事务对数据 a + 500 操作,并提交事。后 A 事务 又对 数据 a 进行查询 a = 1500 。
幻读:事务 A 将数据库中所有数据类型从默认的 true 改成 false,但是事务 B 就在这个时候插入了一条新记录,当事务 A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。
小结:不可重复读的和幻读很容易混淆,不可重复读侧重于修改,幻读侧重于新增或删除。解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表。
事务的四大特性(ACID)
4.1、原子性(Atomicity)
原子性是指事务是一个不可分割的工作单位,事务中的操作要么全部成功,要么全部失败。比如在同一个事务中的SQL语句,要么全部执行成功,要么全部执行失败
4.2、一致性(Consistency)
官网上事务一致性的概念是:事务必须使数据库从一个一致性状态变换到另外一个一致性状态。以转账为例子,A向B转账,假设转账之前这两个用户的钱加起来总共是2000,那么A向B转账之后,不管这两个账户怎么转,A用户的钱和B用户的钱加起来的总额还是2000,这个就是事务的一致性。
4.3、隔离性(Isolation)
事务的隔离性是多个用户并发访问数据库时,数据库为每一个用户开启的事务,不能被其他事务的操作数据所干扰,多个并发事务之间要相互隔离。
4.4、持久性(Durability)
持久性是指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来即使数据库发生故障也不应该对其有任何影响
事务的四大特性中最麻烦的是隔离性,下面重点介绍一下事务的隔离级别
事务的隔离级别
多个线程开启各自事务操作数据库中数据时,数据库系统要负责隔离操作,以保证各个线程在获取数据时的准确性。
5.1、事务不考虑隔离性可能会引发的问题
如果事务不考虑隔离性,可能会引发如下问题:
1、脏读
脏读指一个事务读取了另外一个事务未提交的数据。
这是非常危险的,假设A向B转帐100元,对应sql语句如下所示
1.update account set money=money+100 where name=‘B’;
2.update account set money=money-100 where name=‘A’;
当第1条sql执行完,第2条还没执行(A未提交时),如果此时B查询自己的帐户,就会发现自己多了100元钱。如果A等B走后再回滚,B就会损失100元。
2、不可重复读
不可重复读指在一个事务内读取表中的某一行数据,多次读取结果不同。
例如银行想查询A帐户余额,第一次查询A帐户为200元,此时A向帐户内存了100元并提交了,银行接着又进行了一次查询,此时A帐户为300元了。银行两次查询不一致,可能就会很困惑,不知道哪次查询是准的。
不可重复读和脏读的区别是,脏读是读取前一事务未提交的脏数据,不可重复读是重新读取了前一事务已提交的数据。
很多人认为这种情况就对了,无须困惑,当然是后面的为准。我们可以考虑这样一种情况,比如银行程序需要将查询结果分别输出到电脑屏幕和写到文件中,结果在一个事务中针对输出的目的地,进行的两次查询不一致,导致文件和屏幕中的结果不一致,银行工作人员就不知道以哪个为准了。
3、虚读(幻读)
虚读(幻读)是指在一个事务内读取到了别的事务插入的数据,导致前后读取不一致。
如丙存款100元未提交,这时银行做报表统计account表中所有用户的总额为500元,然后丙提交了,这时银行再统计发现帐户为600元了,造成虚读同样会使银行不知所措,到底以哪个为准。
5.2、事务隔离性的设置语句
MySQL数据库共定义了四种隔离级别:
- Serializable(串行化):可避免 **脏读、不可重复读、虚读(幻读)**情况的发生。
- Repeatable read(可重复读) - 默认的隔离级别:可避免脏读、不可重复读情况的发生。
- Read committed(读已提交):可避免 脏读情况发生。
- Read uncommitted(读未提交):最低级别,以上情况均无法保证。
mysql数据库查询当前事务隔离级别:select @@tx_isolation
*例如:*
mysql数据库默认的事务隔离级别是:Repeatable read(可重复读)
mysql数据库设置事务隔离级别:set transaction isolation level 隔离级别名
在java代码中使用:
Connection connect = JdbcUtils.getConnection();//获取当前数据库的隔离级别System.out.println(connect.getTransactionIsolation());//设置数据库的隔离级别connect.setTransactionIsolation(Connection.TRANSACTION_READ_COMMITTED);//取消事务的自动提交connect.setAutoCommit(false);
事务隔离级别如何在java代码中使用
链接:【JDBC】ACID、四种隔离级别与Java代码实现【附源码】_java代码如何查看oracle当前事务隔离级别-CSDN博客
用户AA为用户BB转账,如果未考虑事务,可能会导致数据的不一致状态。
(1)未考虑事务之前的转账操作:
(此时若没有异常,则转账会成功,若有异常出现,转账操作就会在还没成功之前被迫结束,导致不一致)
//未考虑事务的转账操作,用户AA给用户BB转账100
@Test
public void test01(){String sql = "update user_table set balance = balance - 100 where user = ? ";updateTable(sql,"AA");
// System.out.println(10/0);//模拟转账过程中出现的异常String sql1 = "update user_table set balance = balance + 100 where user = ? ";updateTable(sql1,"BB");
}
public void updateTable(String sql,Object...args) {Connection connect = null;PreparedStatement ps = null;try {//1.获取数据库的连接connect = JdbcUtils.getConnection();//2.预编译sql语句,返回prepareStatement的实例ps = connect.prepareStatement(sql);//3.填充占位符for(int i=0;i<args.length;i++){ps.setObject(i+1,args[i]);}//4.执行sqlps.execute();} catch (Exception e) {e.printStackTrace();} finally {//5.关闭资源JdbcUtils.closeConnection(ps,connect);}
}
(2)考虑事务之后的转账操作:
(我们取消数据库的默认提交操作,当转账操作成功结束时,对结果手动进行提交;若转账未结束时出现异常,我们可以对转账事务进行回滚操作,不管成功与否,数据总是一致的)
public void updateTable(Connection connect, String sql,Object...args) {PreparedStatement ps = null;try {//1.预编译sql语句,返回prepareStatement的实例ps = connect.prepareStatement(sql);//2.填充占位符for(int i=0;i<args.length;i++){ps.setObject(i+1,args[i]);}//3.执行sqlps.execute();} catch (Exception e) {e.printStackTrace();} finally {//4.关闭资源JdbcUtils.closeConnection(ps,null);}
}
//考虑事务之后的转账操作,用户AA给用户BB转账100
@Test
public void test02(){Connection connect = null;try {connect = JdbcUtils.getConnection();String sql = "update user_table set balance = balance - 100 where user = ? ";//1.取消事务的自动提交connect.setAutoCommit(false);updateTable(connect,sql,"AA");System.out.println(10/0);//模拟事务处理过程中出现的异常String sql1 = "update user_table set balance = balance + 100 where user = ? ";updateTable(connect,sql1,"BB");//2.事务正常结束后的提交connect.commit();} catch (Exception e) {e.printStackTrace();try {//3.事务操作过程中出现异常时的回滚操作connect.rollback();} catch (SQLException throwables) {throwables.printStackTrace();}} finally {if (connect!=null){try {connect.close();} catch (SQLException throwables) {throwables.printStackTrace();}}}
}
若此时Connection没有被关闭,还可能被重复使用,则需要恢复其自动提交状态,主要针对数据库连接池时的操作。
setAutoCommit(true)。尤其是在使用数据库连接池技术时,执行close()方法前,建议恢复自动提交状态。
Java代码演示及隔离级别的设置
数据库中事务的隔离级别设置结束后,若重启数据库的服务,则所有的数据库设置都会变成默认。
/*** @author wds* @date 2021-12-14 9:38*/
public class TransactionTest01 {//模拟事务A对数据库中数据的查询操作@Testpublic void testTransactionSelect() throws Exception{Connection connect = JdbcUtils.getConnection();//获取当前数据库的隔离级别System.out.println(connect.getTransactionIsolation());//设置数据库的隔离级别connect.setTransactionIsolation(Connection.TRANSACTION_READ_COMMITTED);//取消事务的自动提交connect.setAutoCommit(false);String sql = "select user,password,balance from user_table where user = ?";PreparedStatement ps = connect.prepareStatement(sql);User user = queryForTable(connect, User.class, sql, "CC");System.out.println(user);System.out.println(connect.getTransactionIsolation());}//模拟事务B对数据库中数据的修改操作@Testpublic void testTransactionUpdate() throws Exception {Connection connect = JdbcUtils.getConnection();//取消事务的自动提交connect.setAutoCommit(false);String sql = "update user_table set balance = ? where user = ?";updateTable(connect,sql,50000,"CC");//线程休眠15秒Thread.sleep(15000);System.out.println("修改结束...");}//考虑事务之后的通用的增删改操作public void updateTable(Connection connect, String sql,Object...args) {PreparedStatement ps = null;try {//1.预编译sql语句,返回prepareStatement的实例ps = connect.prepareStatement(sql);//2.填充占位符for(int i=0;i<args.length;i++){ps.setObject(i+1,args[i]);}//3.执行sqlps.execute();} catch (Exception e) {e.printStackTrace();} finally {//4.关闭资源JdbcUtils.closeConnection(ps,null);}}//考虑事务之后不同数据表的通用的单行查询操作public <T> T queryForTable(Connection connect, Class<T> clazz, String sql,Object...args) {PreparedStatement ps = null;ResultSet resultSet = null;try {//1.预编译Sqlps = connect.prepareStatement(sql);for(int i=0;i<args.length;i++){ps.setObject(i+1,args[i]);}//2.执行sql操作resultSet = ps.executeQuery();//查询结果集的元数据ResultSetMetaData metaData = resultSet.getMetaData();//查询结果集的列数int columnCount = metaData.getColumnCount();if(resultSet.next()){//newInstance()只能调用无参构造方法,创建当前类的对象T t = clazz.newInstance();//返回结果集中的每一个列for(int i=0;i<columnCount;i++){//获取每个列的列值,通过resultSetObject columnValue = resultSet.getObject(i + 1);//通过ResultSetMetaData//获取每个列的列名String columnName = metaData.getColumnName(i + 1);//获取每个列的别名String columnLabel = metaData.getColumnLabel(i + 1);//通过反射,将对象指定名getColumnName的属性设置为指定的属性值:columnValueField field = clazz.getDeclaredField(columnLabel);field.setAccessible(true);field.set(t,columnValue);}return t;}} catch (Exception e) {e.printStackTrace();} finally {JdbcUtils.closeConnection(ps,null,resultSet);}return null;}
}
5.3、使用MySQL数据库演示不同隔离级别下的并发问题
同时打开两个窗口模拟2个用户并发访问数据库
1、当把事务的隔离级别设置为read uncommitted时,会引发脏读、不可重复读和虚读
A窗口
set transaction isolation level read uncommitted; --设置A用户的数据库隔离级别为Read uncommitted(读未提交)
start transaction; --开启事务
select * from account; --查询A账户中现有的钱,转到B窗口进行操作
select * from account --发现a多了100元,这时候A读到了B未提交的数据(脏读)
B窗口
start transaction; --开启事务
update account set money=money+100 where name=‘A’;–不要提交,转到A窗口查询
2、当把事务的隔离级别设置为read committed时,会引发不可重复读和虚读,但避免了脏读
A窗口
set transaction isolation level read committed;
start transaction;
select * from account;–发现a帐户是1000元,转到b窗口
select * from account;–发现a帐户多了100,这时候,a读到了别的事务提交的数据,两次读取a帐户读到的是不同的结果(不可重复读)
B窗口
start transaction;
update account set money=money+100 where name=‘aaa’;
commit;–转到a窗口
3、当把事务的隔离级别设置为repeatable read(mysql默认级别)时,会引发虚读,但避免了脏读、不可重复读
A窗口
set transaction isolation level repeatable read;
start transaction;
select * from account;–发现表有4个记录,转到b窗口
select * from account;–可能发现表有5条记录,这时候发生了a读取到另外一个事务插入的数据(虚读)
B窗口
start transaction;
insert into account(name,money) values(‘ggg’,1000);
commit;–转到a窗口
4、当把事务的隔离级别设置为Serializable时,会避免所有问题
A窗口
set transaction isolation level Serializable;
start transaction;
select * from account; --转到b窗口
B窗口
start transaction;
insert into account(name,money) values(‘ggg’,1000); --发现不能插入,只能等待a结束事务才能插入
相关文章:
数据库的事务的基本特性,事务的隔离级别,事务隔离级别如何在java代码中使用,使用MySQL数据库演示不同隔离级别下的并发问题
文章目录 数据库的事务的基本特性事务的四大特性(ACID)4.1、原子性(Atomicity)4.2、一致性(Consistency)4.3、隔离性(Isolation)4.4、持久性(Durability) 事务的隔离级别5.1、事务不…...
Robust taboo search for the quadratic assignment problem-二次分配问题的鲁棒禁忌搜索
文章目录 摘要关键字结论研究背景1. Introduction 常用基础理论知识2. The quadratic assignment problem3. Taboo search3.1. Moves3.2 Taboo list3.3. Aspiration function3.4. Taboo list size4. Random problems5. Parallel taboo search 研究内容、成果7. Conclusion 潜在…...
Linux:创建进程 -- fork,到底是什么?
相信大家在初学进程时,对fork函数创建进程一定会有很多的困惑,比如: 1.fork做了什么事情?? 2.为什么fork函数会有两个返回值?3.为什么fork的两个返回值,会给父进程谅回子进程pid,给子进程返回0?4.fork之后:父子进…...
基于SpringBoot+vue的token验证
后端: 1,写一个验证token的拦截器 import com.fasterxml.jackson.databind.ObjectMapper; import com.ffyc.news.model.CommonData; import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest; impor…...
Clickhouse设置多磁盘存储策略
设置多磁盘存储 clickhouse安装完成以后,配置了一个默认的存储空间, 这个只能配置一个目录,如果要使用多个磁盘目录,则需要配置磁盘组策略 查看当前的存储策略 select name, path, formatReadableSize(free_space) as free, fo…...
Python开发运维:Django 4.2.7 使用Celery 5.3.5 完成异步和定时任务
目录 一、实验 1.Django使用Celery完成异步和定时任务 二、问题 1. 如何查看Django版本 一、实验 1.Django使用Celery完成异步和定时任务 (1)安装Django (2)新建Django项目 (3)初始框架 (4)urls.py引用视图views from django.contrib import admin from django.urls imp…...
媒体增加日活量的有效策略
随着数字媒体的蓬勃发展,提高日活量成为媒体平台追求的重要目标之一。日活量的增加不仅意味着更广泛的影响力,还能为媒体平台带来更多的商业机会。以下是一些有效的策略,可帮助媒体提高日活量: admaoyan猫眼聚合 内容优质化&#…...
es6新特性总结
1、支持了let和const,为了防止var声明变量带来的变量提升 (1)、存在块级作用域不存在变量提升(考虑暂时性死区) (2)、不允许重复声明(包括普通变量和函数参数)变量提升…...
Spring Boot + hutool 创建海报图片
Spring Boot hutool 创建海报图片 /*** 分享,生成图片* param id* return*/GetMapping("/getShareImg")public void getShareImg(String id,HttpServletResponse response) throws IOException {CouponConsignSaleClassify byId couponConsignSaleClassifyService…...
0002Java程序设计-springboot在线考试系统小程序
文章目录 **摘 要****目录**系统实现开发环境 编程技术交流、源码分享、模板分享、网课分享 企鹅🐧裙:776871563 摘 要 本毕业设计的内容是设计并且实现一个基于springboot的在线考试系统小程序。它是在Windows下,以MYSQL为数据库开发平台&…...
Linux(Centos)上使用crontab实现定时任务(定时执行脚本)
场景 Windows中通过bat定时执行命令和mysqldump实现数据库备份: Windows中通过bat定时执行命令和mysqldump实现数据库备份_mysqldump bat-CSDN博客 上面讲windows中使用bat实现定时任务的方式,如果是在linux上可以通过crontab实现。 cron是服务名称。…...
【Leetcode合集】20. 有效的括号
20. 有效的括号 20. 有效的括号 代码仓库地址: https://github.com/slience-me/Leetcode 个人博客 :https://slienceme.xyz 给定一个只包括 (,),{,},[,] 的字符串 s ,判断字符串…...
OpenGL 绘制线(Qt)
文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 这里同样对OpenGL中的绘制线操作进行封装,便于后续的操作,很多形状也都是基于线来生成的,如圆形等。 二、实现代码 LineDrawable.h #ifndef LINE_DRAWABLE_H #define LINE_DRAWABLE_H#include...
Java | 多线程并发编程CountDownLatch实践
关注:CodingTechWork 引言 在一次数据割接需求中,数据需要通过编程的方式进行转移割接到新平台,此时若串行化方式,无疑会拉锯此次战斗,所以首当其冲要使用并发编程来降低割接时长。 本次主要考虑使用CountDownLatc…...
分布式定时任务系列6:XXL-job触发日志过大引发的CPU告警
传送门 分布式定时任务系列1:XXL-job安装 分布式定时任务系列2:XXL-job使用 分布式定时任务系列3:任务执行引擎设计 分布式定时任务系列4:任务执行引擎设计续 分布式定时任务系列5:XXL-job中blockingQueue的应用 …...
Spark RDD、DataFrame和Dataset的区别和联系
一、三种数据介绍 是Spark中的三种不同的数据结构,它们都可以用于分布式数据处理,但是它们的实现方式和使用方法略有不同。 RDD(弹性分布式数据集) RDD是Spark最初的核心数据结构,它是一个分布式的、只读的、可容错的…...
代码随想录算法训练营第四十五天|139.单词拆分、背包问题总结
LeetCode 139. 单词拆分 题目链接:139. 单词拆分 - 力扣(LeetCode) 这道题使用完全背包来实现,我们首先考虑字符串是否可以由字符串列表组成,因此dp数组大小为n 1 ,其意义是,在n个位置时是否能…...
深度学习卫星遥感图像检测与识别 -opencv python 目标检测 计算机竞赛
文章目录 0 前言1 课题背景2 实现效果3 Yolov5算法4 数据处理和训练5 最后 0 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 **深度学习卫星遥感图像检测与识别 ** 该项目较为新颖,适合作为竞赛课题方向,学长非常推荐…...
wxWidgets 3.2.4发布 —— 发布于2023年11月11日
稳定的3.2系列中的另一个版本现在可以在GitHub上获得。您可以在那里找到包含库源代码和文档的归档文件,以及所选Windows编译器(如Microsoft Visual C、MinGW-w64和TDM-GCC)的二进制文件。您还可以阅读此版本的更新文档,特别是&…...
PyQt6运行QTDesigner生成的ui文件程序
2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~共计18条视频,包括:2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~、第2讲 PyQt6库和工具库Q…...
MFC内存泄露
1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院挂号小程序
一、开发准备 环境搭建: 安装DevEco Studio 3.0或更高版本配置HarmonyOS SDK申请开发者账号 项目创建: File > New > Create Project > Application (选择"Empty Ability") 二、核心功能实现 1. 医院科室展示 /…...
oracle与MySQL数据库之间数据同步的技术要点
Oracle与MySQL数据库之间的数据同步是一个涉及多个技术要点的复杂任务。由于Oracle和MySQL的架构差异,它们的数据同步要求既要保持数据的准确性和一致性,又要处理好性能问题。以下是一些主要的技术要点: 数据结构差异 数据类型差异ÿ…...
【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...
selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...
《C++ 模板》
目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板,就像一个模具,里面可以将不同类型的材料做成一个形状,其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式:templa…...
[USACO23FEB] Bakery S
题目描述 Bessie 开了一家面包店! 在她的面包店里,Bessie 有一个烤箱,可以在 t C t_C tC 的时间内生产一块饼干或在 t M t_M tM 单位时间内生产一块松糕。 ( 1 ≤ t C , t M ≤ 10 9 ) (1 \le t_C,t_M \le 10^9) (1≤tC,tM≤109)。由于空间…...
RushDB开源程序 是现代应用程序和 AI 的即时数据库。建立在 Neo4j 之上
一、软件介绍 文末提供程序和源码下载 RushDB 改变了您处理图形数据的方式 — 不需要 Schema,不需要复杂的查询,只需推送数据即可。 二、Key Features ✨ 主要特点 Instant Setup: Be productive in seconds, not days 即时设置 :在几秒钟…...
篇章一 论坛系统——前置知识
目录 1.软件开发 1.1 软件的生命周期 1.2 面向对象 1.3 CS、BS架构 1.CS架构编辑 2.BS架构 1.4 软件需求 1.需求分类 2.需求获取 1.5 需求分析 1. 工作内容 1.6 面向对象分析 1.OOA的任务 2.统一建模语言UML 3. 用例模型 3.1 用例图的元素 3.2 建立用例模型 …...
解密鸿蒙系统的隐私护城河:从权限动态管控到生物数据加密的全链路防护
摘要 本文以健康管理应用为例,展示鸿蒙系统如何通过细粒度权限控制、动态权限授予、数据隔离和加密存储四大核心机制,实现复杂场景下的用户隐私保护。我们将通过完整的权限请求流程和敏感数据处理代码,演示鸿蒙系统如何平衡功能需求与隐私安…...
