事务:事务是并发控制的基本单位。是指逻辑上的一组操作,一组SQL语句,组成这组操作的各个功能要么全部成功,要么全部不成功。
有关事务的几个术语如下:
Start transaction 开启事务
rollback 回退 指撤销指定SQL语句的过程
commit 提交事务 指将未存储的SQL语句结果写入数据库表
savepoint 保留点,指事务处理中设置的临时占位符(place-holder),你可以对它发布回退(与回退整个事务处理不同)
Select * from order;
Start transaction;
Delete from order;
Select * from order;
Rollback;
Select * from order;
在事务处理块中,提交不会隐含地进行。为进行明确的提交,使用 COMMIT 语句,如下所示:
Start transaction
Delete from order where order_num =20010;
Delete from order where order_num=20010;
Commit;
设置保留点:
Savepoint delete1;
每个保留点都取标识它的唯一名字,以便在回退时,MySQL知道要回退到何处。为了回退到本例给出的保留点,可如下进行:
Rollback to delete1;
更改默认提交行为:
Set autocommit =0;
autocommit 标志决定是否自动提交更改,不管有没有 COMMIT语句。设置 autocommit 为 0 (假)指示MySQL不自动提交更改(直到 autocommit 被设置为真为止)。
JDBC中使用事务
当Jdbc程序向数据库获得一个Connection对象时,默认情况下这个Connection对象会自动向数据库提交在它上面发送的SQL语句。若想关闭这种默认提交方式,让多条SQL在一个事务中执行,可使用下列的JDBC控制事务语句
事务的四大特性:
原子性:指的是事务是一个不可分割的工作单位,事务中的操作要么全部成功,要么全部失败,比如在同一个事务中的SQL语句,要么全部执行成功,要么全部执行失败。
一致性:事务必须使数据库从一个一致性状态变换到另外一个一致性状态。以转账为例,A向B转账,假设转账之前A B两用户总额为200,那么A向B转账之后,不管这两个账户怎么转,AB两用户的总额不变还是200,这个就是事务的一致性。
隔离性:指的是多个用户并发访问数据库时,数据库为每一个用户开启事务,不能被其他的事务操作所干扰,多个并发事务之间要相互隔离。
即要达到这么一种效果:对于任意两个并发的事务T1和T2,在事务T1看来,T2要么在T1开始之前就已经结束,要么在T1结束之后才开始,这样每个事务都感觉不到有其他事务在并发地执行。
持久性:个事务一旦被提交。对数据库的改变就是永久性的,即使数据库发生故障也不会造成影响。
例如我们在使用JDBC操作数据库时,在提交事务方法后,提示用户事务操作完成,当我们程序执行完成直到看到提示后,就可以认定事务以及正确提交,即使这时候数据库出现了问题,也必须要将我们的事务完全执行完成,否则就会造成我们看到提示事务处理完毕,但是数据库因为故障而没有执行事务的重大错误。
如果事务不考虑隔离性,可能会引发如下问题:
1、脏读
脏读指一个事务读取了另外一个事务未提交的数据。
这是非常危险的,假设A向B转帐100元,对应sql语句如下所示
1.update account set money=money+100 where name='B'; (此时A通知B)
2.update account set money=money-100 where name='A';
当只执行第一条SQL时,A通知B查看账户,B发现确实钱已到账(此时即发生了脏读),而之后无论第二条SQL是否执行,只要该事务不提交,则所有操作都将回滚,那么当B以后再次查看账户时就会发现钱其实并没有转。
2、不可重复读
不可重复读指在一个事务内读取表中的某一行数据,多次读取结果不同。
例如银行想查询A帐户余额,第一次查询A帐户为200元,此时A向帐户内存了100元并提交了,银行接着又进行了一次查询,此时A帐户为300元了。银行两次查询不一致,可能就会很困惑,不知道哪次查询是准的。
不可重复读和脏读的区别是,脏读是某一事务读取了另一个事务未提交的脏数据,而不可重复读则是读取了前一事务提交的数据。
在某些情况下,不可重复读并不是问题,比如我们多次查询某个数据当然以最后查询得到的结果为主。但在另一些情况下就有可能发生问题,例如对于同一个数据A和B依次查询就可能不同,A和B就可能打起来了……
3、虚读(幻读)
幻读是事务非独立执行时发生的一种现象。例如事务T1对一个表中所有的行的某个数据项做了从“1”修改为“2”的操作,这时事务T2又对这个表中插入了一行数据项,而这个数据项的数值还是为“1”并且提交给数据库。而操作事务T1的用户如果再查看刚刚修改的数据,会发现还有一行没有修改,其实这行是从事务T2中添加的,就好像产生幻觉一样,这就是发生了幻读。
幻读和不可重复读都是读取了另一条已经提交的事务(这点就脏读不同),所不同的是不可重复读查询的都是同一个数据项,而幻读针对的是一批数据整体(比如数据的个数)。
事务隔离性的设置语句
MySQL数据库共定义了四种隔离级别:
mysql数据库查询当前事务隔离级别:select @@tx_isolation
以上四种隔离级别最高的是Serializable级别,最低的是Read uncommitted级别,当然级别越高,执行效率就越低。像Serializable这样的级别,就是以锁表的方式(类似于Java多线程中的锁)使得其他的线程只能在锁外等待,所以平时选用何种隔离级别应该根据实际情况。在MySQL数据库中默认的隔离级别为Repeatable read (可重复读)。
参考资料:
http://www.cnblogs.com/xdp-gacl/p/3984001.html
https://www.cnblogs.com/fjdingsd/p/5273008.html
Sql语句中IN和exists的区别及应用:
https://www.cnblogs.com/liyasong/p/sql_in_exists.html
SQL的几种连接:内连接、左联接、右连接、全连接、交叉连接
https://www.cnblogs.com/zxlovenet/p/4005256.html
视图和表的区别
表与视图很相似,都是可以包含相同类型的数据的二维结构,都有行,列,单元格,在select语句的from字句中,都可以将他们用作数据源。
当需要区分表和视图的时候,通常表被称为基表或者数据表。
不同之处:
a:表直接将数据存储在磁盘上,视图是将sql语句存储到磁盘上
b:视图是建立在表的基础上,表存储数据库中的数据,而视图显示已经在表中的数据的外观
c:表是静态的,而视图是动态的,意思是表中数据发生了改变,其建立在表基础的视图跟着改变
d:通过视图不能改变表中数据。
e:删除视图,表不受影响,而删除表,视图不再起作用
f:视图本身没有数据,只保存了sql语句
总之:记住视图只保存了sql语句,没有保存数据,每次使用视图时会去执行sql语句在它的基表中查询数据,而表却是实实在在的保存着数据。可以将查询语句保存到视图中,在每次需要数据的时候去执行查询,也可以直接将查询语句的结果保存到一个表中,以后使用的时候不再查询,前者没有后者快,后者比前者需要更多的磁盘空间。
可以通过视图来修改基表的数据,但是有很多限制。只支持单表修改
触发器
https://blog.csdn.net/weixin_39941298/article/details/81080353