【FMDB】事务

面试的时候会遇到这样问题,一边写入和一边读取一条数据,怎么保证数据的准确性?这里就是问事务了。

影像数据库的常见因素

为了节省时间,我大部分还是借鉴别人的博客,主要为了记录一些问题

下文摘此文 http://blog.csdn.net/x32sky/article/details/45531229

1.事务(Transaction)是并发控制的基本单位。所谓的事务,它是一个操作序列,这些操作要么都执行,要么都不执行,它是一个不可分割的工作单位。例如,银行转账工作:从一个账号扣款并使另一个账号增款,这两个操作要么都执行,要么都不执行。所以,应该把它们看成一个事务。事务是数据库维护数据一致性的单位,在每个事务结束时,都能保持数据一致性。

针对上面的描述可以看出,事务的提出主要是为了解决并发情况下保持数据一致性的问题。

事务具有以下4个基本特征。

●   Atomic(原子性):事务中包含的操作被看做一个逻辑单元,这个逻辑单元中的操作要么全部成功,要么全部失败。

●   Consistency(一致性):只有合法的数据可以被写入数据库,否则事务应该将其回滚到最初状态。

●   Isolation(隔离性):事务允许多个用户对同一个数据进行并发访问,而不破坏数据的正确性和完整性。同时,并行事务的修改必须与其他并行事务的修改相互独立。

●   Durability(持久性):事务结束后,事务处理的结果必须能够得到固化。

2.事务的语句
 开始事物:BEGIN TRANSACTION
 提交事物:COMMIT TRANSACTION
 回滚事务:ROLLBACK TRANSACTION
3.事务的4个属性
     ①原子性(Atomicity):事务中的所有元素作为一个整体提交或回滚,事务的个元素是不可分的,事务是一个完整操作。
  ②一致性(Consistemcy):事物完成时,数据必须是一致的,也就是说,和事物开始之前,数据存储中的数据处于一致状态。保证数据的无损。
  ③隔离性(Isolation):对数据进行修改的多个事务是彼此隔离的。这表明事务必须是独立的,不应该以任何方式以来于或影响其他事务。
  ④持久性(Durability):事务完成之后,它对于系统的影响是永久的,该修改即使出现系统故障也将一直保留,真实的修改了数据库
4.事务的保存点
     SAVE TRANSACTION 保存点名称 --自定义保存点的名称和位置
     ROLLBACK TRANSACTION 保存点名称 --回滚到自定义的保存点

其他高手的一些补充:

事务的标准定义: 指作为单个逻辑工作单元执行的一系列操作,而这些逻辑工作单元需要具有原子性,  一致性,隔离性和持久性四个属性,统称为ACID特性。

所谓事务是用户定义的一个数据库操作序列,这些操作要么全做要么全不做,是一个不可分割的工作单位。例如,在关系数据库中,一个事务可以是一条SQL语句、一组SQL语句或整个程序。 
事务和程序是两个概念。一般地讲,一个程序中包含多个事务。
事务的开始与结束可以由用户显式控制。如果用户没有显式地定义事务,则由DBMS按缺省规定自动划分事 
务。在SQL语言中,定义事务的语句有三条: 
BEGIN TRANSACTION 
COMMIT 
ROLLBACK

同生共死。。
显示事务被用begin transaction 与 end transaction 标识起来,其中的 update 与 delete 语句或者全部执行或者全部不执行。。 如:
begin transaction T1
update student
set name=‘Tank‘
where id=2006010
delete from student
where id=2006011
commit
简单地说,事务是一种机制,用以维护数据库的完整性。

其实现形式就是将普通的SQL语句嵌入到Begin Tran...Commit Tran 中(或完整形式 Begin Transaction...Commit Transaction),当然,必要时还可以使用RollBack Tran 回滚事务,即撤销操作。

利用事务机制,对数据库的操作要么全部执行,要么全部不执行,保证数据库的一致性。需要使用事务的SQL语句通常是更新和删除操作等。

end transaction T1

关于savepoint

用户在事务(transaction)内可以声明(declare)被称为保存点(savepoint)
的标记。保存点将一个大事务划分为较小的片断。

用户可以使用保存点(savepoint)在事务(transaction)内的任意位置作标
记。之后用户在对事务进行回滚操作(rolling back)时,就可以选择从当前
执行位置回滚到事务内的任意一个保存点。例如用户可以在一系列复杂的更
新(update)操作之间插入保存点,如果执行过程中一个语句出现错误,用
户 可以回滚到错误之前的某个保存点,而不必重新提交所有的语句。

在开发应用程序时也同样可以使用保存点(savepoint)。如果一个过程
(procedure)内包含多个函数(function),用户可以在每个函数的开始位置
创建一个保存点。当一个函数失败时, 就很容易将数据恢复到函数执行之前
的状态,回滚(roll back)后可以修改参数重新调用函数,或执行相关的错误
处理。

当事务(transaction)被回滚(rollback)到某个保存点(savepoint)后,
Oracle将释放由被回滚语句使用的锁。其他等待被锁资源的事务就可以继续
执行。需要更新(update)被锁数据行的事务也可以继续执行。

将事务(transaction)回滚(roll back)到某个保存点(savepoint)的过程如
下:
1. Oracle 回滚指定保存点之后的语句
2. Oracle 保留指定的保存点,但其后创建的保存点都将被清除
3. Oracle 释放此保存点后获得的表级锁(table lock)与行级锁(row
lock),但之前的数据锁依然保留。

被部分回滚的事务(transaction)依然处于活动状态,可以继续执行。

一个事务(transaction)在等待其他事务的过程中,进行回滚(roll back)到
某个保存点(savepoint)的操作不会释放行级锁(row lock)。为了避免事务
因为不能获得锁而被挂起,应在执行 UPDATE 或 DELETE 操作前使用 FOR
UPDATE ... NOWAIT 语句。(以上内容讲述的是回滚保存点之前所获得的
锁。而在保存点之后获得的行级锁是会被释放的,同时保存点之后执行的
SQL 语句也会被完全回滚)。

事务处理机制不但有以上特性,原子,一致,隔离,持久,还有性能上也有不错的表现,以下是测试使用事务和不使用事务的测试代码

运行后可以很明显的看出性能上的区别。

[objc] view plaincopyprint?

    1. -(BOOL)openDB
    2. {
    3. BOOL isFile = NO;
    4. NSString *DBFilePath = [self getDBPath];
    5. NSFileManager *filaManager = [NSFileManager defaultManager];
    6. if([filaManager fileExistsAtPath:DBFilePath])
    7. isFile = YES;
    8. self.iFmDb = [FMDatabase databaseWithPath:DBFilePath];
    9. NSLog(@"openDB");
    10. if (![self.iFmDb open])
    11. {
    12. return NO;
    13. }
    14. return YES;
    15. }
    16. -(void)testDBSpeed
    17. {
    18. NSDate *date1 = [NSDate date];
    19. [self insertData:500 useTransaction:NO];
    20. NSDate *date2 = [NSDate date];
    21. NSTimeInterval a = [date2 timeIntervalSince1970] - [date1 timeIntervalSince1970];
    22. NSLog(@"不使用事务插入500条数据用时%.3f秒",a);
    23. [self insertData:1000 useTransaction:YES];
    24. NSDate *date3 = [NSDate date];
    25. NSTimeInterval b = [date3 timeIntervalSince1970] - [date2 timeIntervalSince1970];
    26. NSLog(@"使用事务插入500条数据用时%.3f秒",b);
    27. }
    28. - (void)insertData:(int)fromIndex useTransaction:(BOOL)useTransaction
    29. {
    30. [self openDB];
    31. if (useTransaction) {
    32. [self.iFmDb beginTransaction];
    33. BOOL isRollBack = NO;
    34. @try {
    35. for (int i = fromIndex; i<500+fromIndex; i++) {
    36. NSString *nId = [NSString stringWithFormat:@"%d",i];
    37. NSString *phone= [[NSString alloc] initWithFormat:@"phone_%d",i];
    38. NSString *strName = [[NSString alloc] initWithFormat:@"name_%d",i];
    39. NSString *roomID= [[NSString alloc] initWithFormat:@"roomid_%d",i];
    40. NSString *sql = @"INSERT INTO GroupPersonInfo(uid,phone,name,groupRoomId) VALUES (?,?,?,?)";
    41. BOOL a = [self.iFmDb executeUpdate:sql,nId,phone,strName,roomID];
    42. if (!a) {
    43. NSLog(@"插入失败1");
    44. }
    45. }
    46. }
    47. @catch (NSException *exception) {
    48. isRollBack = YES;
    49. [self.iFmDb rollback];
    50. }
    51. @finally {
    52. if (!isRollBack) {
    53. [self.iFmDb commit];
    54. }
    55. }
    56. }else{
    57. for (int i = fromIndex; i<500+fromIndex; i++) {
    58. NSString *nId = [NSString stringWithFormat:@"%d",i];
    59. NSString *phone= [[NSString alloc] initWithFormat:@"phone_%d",i];
    60. NSString *strName = [[NSString alloc] initWithFormat:@"name_%d",i];
    61. NSString *roomID= [[NSString alloc] initWithFormat:@"roomid_%d",i];
    62. NSString *sql = @"INSERT INTO GroupPersonInfo(uid,phone,name,groupRoomId) VALUES (?,?,?,?)";
    63. BOOL a = [self.iFmDb executeUpdate:sql,nId,phone,strName,roomID];
    64. if (!a) {
    65. NSLog(@"插入失败2");
    66. }
    67. }
    68. }
    69. [self closeDB];
    70. }
    71. - (BOOL) closeDB
    72. {
    73. NSLog(@"closeDB");
    74. if (self.iFmDb)
    75. {
    76. return [self.iFmDb close];
    77. }
    78. return NO;
    79. }
时间: 2024-10-12 01:53:47

【FMDB】事务的相关文章

iOS 使用FMDB事务批量更新数据库

当谈到“事务”时,我想到了sqlite本身是支持事务操作的,FMDB作为sqlite的上层封装也对事务进行了支持,那么说了这么多给不了解“事务”这个词的同学解释一下吧:通常一次 sqlite3_exec 就是一次事务,假如你要对数据库中的Stutent表插入新数据,那么该事务的具体过程是:开始新事物->插入数据->提交事务,那么当我们要往该表内插入500条数据,如果按常规操作处理就要执行500次“开始新事物->插入数据->提交事务”的过程. 好吧,今天的重点来了,举个例子:假如北京

FMDB事务的使用

http://blog.csdn.net/qq_29892943/article/details/50541439 首先,说一下事务是什么,比如说我们有一个学生表和一个学生成绩表,而且一个学生对应一个学生成绩.比如小明的成绩是100分,那么我们要写两个sql语句对不同的表进行插入数据.但是如果在这个过程中,小明这个学生成功的插入到数据库,而成绩插入时失败了,怎么办?这时事务就突出了它的作用.用事务可以对两个表进行同时插入,一旦一个表插入失败,那么就会进行事务回滚,就是让另一个表也不进行插入数据了

使用FMDB事务批量存入数据

最近学习在iOS编程中使用FMDB来存储数据,在inssert一大串对象时发现用了超过20秒,如果插入1000条数据,相当于进行了1000次磁盘存储.而使用事务之后相当于只执行一次sql语句,插入速度大大提升. [[DBManager sharedDBManager].database beginTransaction]; @try { for (NSString *sql in self.sqlArray) { [[DBManager sharedDBManager].database exe

关于FMDB事务(Transaction)的解读

事务(Transaction)的描述: 事务(Transaction)是并发控制的基本单位.所谓的事务,它是一个操作序列,这些操作要么都执行,要么都不执行,它是一个不可分割的工作单位.例如,银行转账工作:从一个账号扣款并使另一个账号增款,这两个操作要么都执行,要么都不执行.所以,应该把它们看成一个事务.事务是数据库维护数据一致性的单位,在每个事务结束时,都能保持数据一致性.针对上面的描述可以看出,事务的提出主要是为了解决并发情况下保持数据一致性的问题. 事务的标准定义: 指作为单个逻辑工作单元执

使用FMDB框架来加载数据库

1.打开数据库 先要获取沙盒的数据库文件名,并创建数据库文件名,定数据库: @property (nonatomic, strong) FMDatabase *db; // 0.获得沙盒中的数据库文件名 NSString *filename = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"

iOS开发tips总结

tip 1 :  给UIImage添加毛玻璃效果 func blurImage(value:NSNumber) -> UIImage { let context = CIContext(options:[KCIContextUseSoftwareRenderer:true]) let ciImage = CoreImage.CIImage(image:self) let blurFilter = CIFilter(name:"CIGassianBlur") blurFilter?

iOS开发200个tips总结(一)

tip 1 :  给UIImage添加毛玻璃效果 func blurImage(value:NSNumber) -> UIImage { let context = CIContext(options:[KCIContextUseSoftwareRenderer:true]) let ciImage = CoreImage.CIImage(image:self) let blurFilter = CIFilter(name:"CIGassianBlur") blurFilter?

FMDB开启事务

#import "RootViewController.h" #import "FMDatabase.h" @interface RootViewController () { FMDatabase *_dataBase; } @end @implementation RootViewController - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil

FMDB 使用方法

优秀的第三方库,README 也是很优秀的,理解了 README,会对使用带来很多便利. ARC 和 MRC 项目中使用 ARC 还是 MRC,对使用 FMDB 都没有任何影响,FMDB 会在编译项目时自动匹配. 使用 在 FMDB 中有三个重要的类: FMDatabase:是一个提供 SQLite 数据库的类,用于执行 SQL 语句. FMResultSet:用在 FMDatabase 中执行查询的结果的类. FMDatabaseQueue:在多线程下查询和更新数据库用到的类. 数据库创建 F