浅谈数据库--事务(mysql)

事务

  事务其实是一组对数据库增删改操作的组合,可以这样来理解,当你往某个人身上打1000元的时候,在数据库中会发生两个改变,一个是你的钱减少了,另一个是那个人的钱增加了,这两个操作必须同时满足,不然问题就大了,怎样保证两个操作全部执行,这就需要mysql事务的支持。


mysql是支持事务的,但首先确认你是InnoDB存储引擎

mysql事务是为了维护数据库的完整性,堆成批量的语句要么全部执行,要么全部不执行。一般用来管理insert delete 和update语句的。

事务的特点:

1、事务的原子性:一组事务,要么成功;要么撤回。

2、稳定性 : 有非法数据(外键约束之类),事务撤回。

3、隔离性:事务独立运行。一个事务处理后的结果,影响了其他事务,那么其他事务会撤回。事务的100%隔离,需要牺牲速度。

4、可靠性:软、硬件崩溃后,InnoDB数据表驱动会利用日志文件重构修改。可靠性和高速度不可兼得, innodb_flush_log_at_trx_commit选项 决定什么时候吧事务保存到日志里。

下面是在mysql中操作的一些命令;

set autoaction=0;//这条命令用来取消mysql的自动提交。
begin;//事务开始。
savepoint pointname;//设立存储点,设立多个可以使用rollback返回到某一个上。
rollback pointname;//返回到某个point。
rollback;//不使用point则返回到begin。
commit;//如果可以提交则用commit提交。

使用 show variables like ‘autocommit‘;可以查看当前autoaction的状态

下面是我在mysql下执行的测试:

可以看出使用rollback后退回到了添加之前的数据。

使用savepoint的方式,读者可以下去测试。

下面是C语言下对事务的测试:(使用connect c包)

首先测试前表中的数据时这样的

我的测试代码如下:

这里的my_sql.h只给出了关于事务这部分的代码(关于数据库的所有操作的代码会在后面的手动搭建
http服务器的博客中给出)。
my_sql.h   
 13     }
 14     bool HttpSql::mysql_start()
 15     {
 16         if(mysql_query(mysql,"SET autocommit=0")==0)
 17         {
 18             cout<<"start success"<<endl;
 19             return true;
 20         }
 21         else {
 22             return false;
 23         }
 24 
 25     }
 26     bool HttpSql::mysql_begin()
 27     {
 28         if(mysql_query(mysql,"BEGIN")==0)
 29         {
 30             cout<<"begin success"<<endl;
 31             return true;
 32         }
 33         else{
 34             return false;
 35         }
 36     }
 37     bool HttpSql::mysql_commit()
 38     {
 39         if(mysql_query(mysql,"COMMIT")==0)
 40         {
 41             cout<<"commit success"<<endl;
 42             return true;
 43         }
 44         else{
 40         {
 41             cout<<"commit success"<<endl;
 42             return true;
 43         }
 44         else{
 45             return false;
 46         }
 47     }
 48     bool HttpSql::mysql_rollback()
 49     {
 50         if(mysql_query(mysql,"ROLLBACK")==0)
 51         {
 52             cout<<"rollback success"<<endl;
 53             return true;
 54         }
 55         else
 56         {
 57             return false;
 58         }
 59     }
121     bool HttpSql::mysql_modify(string str)
122     {
123         bool ret=false;
124         string _str="update test_info set ";
125     //  cout<<str<<endl;
126         _str+=str.c_str();
127     //  cout<<_str<<endl;
128         ret=mysql_op(_str);
129         if(ret)
130         {
131             return 1;
132         }else{
133             return -1;
134         }  
135      }                                                                                                                              46,3-9        28%                                                                                                                                           32,2-8         8%
modify.cpp

#include"my_sql.h"

104 int main()
105 {
106     HttpSql sql;
107     if(!sql.mysql_start())
108     {
109         return -1;
110     }
111     char buf1[1024];
112     char buf2[1024];
113     memset(buf1,‘\0‘,sizeof(buf1));
114     memset(buf2,‘\0‘,sizeof(buf2));
115     strcpy(buf1,"update test_info set name=\"wangmazi\" where name=\"zhangsan\"");
116     strcpy(buf2,"update test_info set name=\"wangmazi\" where name\"lisi\"");
            //上面这行代码我故意在where name后面少了一个等于号
117     string str1(buf1);
118     string str2(buf2);
119     if(!sql.mysql_begin())
120     {
121         return -1;
122     }
123     int ret1=sql.mysql_op(str1);
124     int ret2=sql.mysql_op(str2);//因为str2一定会执行失败索引返回false.
125     //ret1=false;
126     if(ret1&&ret2)
127     {
128         if(sql.mysql_commit())
129         {
130             cout<<"modify success"<<endl;
131         }
132         else{
133             cout<<"modify failure"<<endl;
134         }
135     }else    //执行else语句使其退回修改之前的样子。
136     {
137         if(sql.mysql_rollback())
138         {
139             cout<<"modify failure roolback success"<<endl;
140         }
141         else{
142             cout<<"boom!!!1"<<endl;
143         }
144     }
145 
146 
147 }
                                                                                                                                                                                                                                                                                       131,2-8      底端                                                                                                                                           116,2-5       89%

测试结果

显然表中没有发生改变。


更改代码将上面的‘=‘加上结果如下

在使用各种语言对事务进行操作的时候要在最后手动关闭连接 mysql_close();

你以为上面的就正确了吗?确实是正确的,因为上面的都是已经配置好的。


在没有弄好之前,花了测试了好久,才发现一个大坑!!听我慢慢道来:

查阅了相关资料,基本上都是在用show engines;查看是否支持innodb存储引擎,但是测试之后却发现根本rollback不了,在代码中显示的是success,但数据库中却是1 worning; 虽然show engines显示的 innodb是yes,但你仍需要添加这段代码:

alter table test_info type=INNODB;不然你有可能永远都测试成功不了

总结:事务对数据库的完整性具有很深的意义。是不可或缺的一部分。关于事务的使用还有很多方面。需要慢慢学习

时间: 2024-08-14 07:35:12

浅谈数据库--事务(mysql)的相关文章

浅谈数据库事务

事务的四大特性(ACID) 原子性 原子性是指事务包含的所有操作要么全部成功,要么全部失败.例小王要向小李转账200元.则账要么转账成功小王账户减200元,小李账户加200元,要么执行失败,两者账户都不动. 一致性 一致性是指事务执行之前和执行之后都必须处于一致性状态. 假设转账前小王和小李的余额之和是2000元,那么不管他们之间进行了多少次转账,他们的余额之和肯定还是2000元. 隔离性 隔离性是当多个事务并发操作数据库时,不能被其他事务的操作所干扰 例小王银行卡只有200元了,他要向小李转账

浅谈数据库框架,见笑,请多指正

浅谈数据库框架,见笑,请多指正 http://weibo.com/p/1001603724746155003486 一友说"插件式存储又割裂了SQL引擎的完整逻辑...总体而言在现有框架下MySQL的优化器没有多大改进的价值". 我们且做个技术分析: 1 插件式框架,可以静态/动态加载组件,方便在同类不同属家的模块间切换,这种设计是良好的. 很多软件的设计都采用了"微内核+插件"这样的方式构筑了强大的应用.如Ecplise生态圈. 2 数据库范围内, MySQL的属

浅谈数据库并发控制 - 锁和 MVCC

在学习几年编程之后,你会发现所有的问题都没有简单.快捷的解决方案,很多问题都需要权衡和妥协,而本文介绍的就是数据库在并发性能和可串行化之间做的权衡和妥协 - 并发控制机制. 如果数据库中的所有事务都是串行执行的,那么它非常容易成为整个应用的性能瓶颈,虽然说没法水平扩展的节点在最后都会成为瓶颈,但是串行执行事务的数据库会加速这一过程:而并发(Concurrency)使一切事情的发生都有了可能,它能够解决一定的性能问题,但是它会带来更多诡异的错误. 引入了并发事务之后,如果不对事务的执行进行控制就会

浅谈数据库去重

关于sql去重,我简单谈一下自己的简介,如果各位有建议或有不明白的欢迎多多指出.推荐网址:www.4-yecao.com 关于sql去重最常见的有两种方式:DISTINCT和ROW_NUMBER(),当然了ROW_NUMBER()除了去重还有很多其他比较重要的功能,一会我给大家简单说说我自己在实际中用到的. 假如有张UserInfo表,如下图: 现在我们要去掉完全重复的数据:SELECT DISTINCT * FROM dbo.UserInfo结果如下图: 但是现在有个新的需求,要把名字为‘张三

浅谈数据库设计技巧(转)

说到数据库,我认为不能不先谈数据结构.1996年,在我初入大学学习计算机编程时,当时的老师就告诉我们说:计算机程序=数据结构+算法.尽管现在的程序开发已由面向过程为主逐步过渡到面向对象为主,但我还是深深赞同8年前老师的告诉我们的公式:计算机程序=数据结构+算法.面向对象的程序开发,要做的第一件事就是,先分析整个程序中需处理的数据,从中提取出抽象模板,以这个抽象模板设计类,再在其中逐步添加处理其数据的函数(即算法),最后,再给类中的数据成员和函数划分访问权限,从而实现封装. 数据库的最初雏形据说源

浅谈数据库设计

浅谈数据库设计 数据库设计的重要性:好的数据库设计有下面的一些作用: 1.首先充分体现系统的需求,数据库是为应用服务的,好的数据库设计应该首先能满足应用系统的业务需求,准确的表达数据间关系. 2.保证数据的准确性和一致性,通过主外键.非空.限制.唯一索引等保证数据的健壮. 3.提高数据的查询效率,通过合理表结构,安排物理存储分区.增加索引等方式,提高数据的读取速度,提高查询效率. 4.有好的扩展性,在必要时能根据需求扩展数据结构. 在系统设计中对数据库的设计应考虑哪些设计原则  数据库是整个软件

浅谈数据库之存储过程

浅谈数据库之存储过程 什么是存储过程 如果你接触过其他的编程语言,那么就好理解了,存储过程就像是方法一样. 竟然他是方法那么他就有类似的方法名,方法要传递的变量和返回结果,所以存储过程有存储过程名有存储过程参数也有返回值. 存储过程的优点:    存储过程的能力大大增强了SQL语言的功能和灵活性. 可保证数据的安全性和完整性. 通过存储过程可以使没有权限的用户在控制之下间接地存取数据库,从而保证数据的安全. 通过存储过程可以使相关的动作在一起发生,从而可以维护数据库的完整性. 在运行存储过程前,

浅谈分布式事务与TX-LCN

最近做项目使用到了分布式事务,下面这篇文章将给大家介绍一下对分布式事务的一些见解,并讲解分布式事务处理框架TX-LCN的执行原理,初学入门,错误之处望各位不吝指正. 什么情况下需要使用分布式事务? 使用的场景很多,先举一个常见的:在微服务系统中,如果一个业务需要使用到不同的微服务,并且不同的微服务对应不同的数据库. 打个比方:电商平台有一个客户下订单的业务逻辑,这个业务逻辑涉及到两个微服务,一个是库存服务(库存减一),另一个是订单服务(订单数加一),示意图如下: 如果在执行这个业务逻辑时没有使用

浅谈数据库的隔离等级

数据库事务的隔离等级,英语叫做 Transaction Isolation Level. 最近在给客户维护项目的时候,对一个表在两个进程中同时做更新和查询时碰到了死锁(DeadLock),数据表里有几百万上千万条记录,上面的处理当时是更新几千条记录, 查询整张表. 这是前提,为了搞明白这个死锁,大概涉及到死锁,隔离等级,锁升级以及Trace跟踪的内容,陆续整理出来,今天说说隔离等级. 概要 有下面3种读取不具合的现象,首先说说关于RDBMS的ACID特性的I(Isolation-隔离性) ·脏读