Android小技巧(二):为ContentProvider添加数据库事务支持

本文介绍了在Android平台上如何利用SQLite数据库的事务处理功能确保数据导入的一致性和安全性。针对已封装成ContentProvider的SQLite数据库,通过覆写applyBatch方法实现事务支持。
转载请注明出处。博客地址: http://blog.youkuaiyun.com/mylzc

介绍:数据库事务是由一组数据库操作序列组成,事务作为一个整体被执行。

事务的原子性:包含在其中的对数据库的操作序列最终要么全部执行,要么全部不执行。当全部执行时,事务对数据库的修改将生效;当全部不执行时,数据库维持原有的状态,不会被修改。

问题:最近在做一个从sdcard导入数据到数据库的功能,当导入失败时,数据库要恢复到导入前的状态。使用数据库事务处理能很好地满足到我们的需求。

我们知道Android平台上使用的sqlite数据库是支持事务处理功能的,实现的代码如下:

[java]  view plain copy
  1. SQLiteDatabase db =mOpenHelper.getWritableDatabase();  
  2.                    db.beginTransaction();//开始事务  
  3. //进行insertdelete update等数据库操作  
  4. db.setTransactionSuccessful();//设置事务标记为Successful  
  5. db.endTransaction();//提交事务  

可是,对于已经封装成ContentProvider的Sqlite我们应该如何让其支持事务处理功能呢?

解决办法:查看ContentProvider的API说明文档,我们惊喜地发现applyBatch(String authority,ArrayList<ContentProviderOperation> operations)这个方法,难道只需要直接使用这个方法就可以实现事务了?

谨慎起见我们先来看看ContentProvider的源码,最后追踪到这个方法:

[java]  view plain copy
  1. public ContentProviderResult[]applyBatch(ArrayList<ContentProviderOperation> operations)  
  2.             throwsOperationApplicationException {  
  3.        final int numOperations = operations.size();  
  4.        final ContentProviderResult[] results = newContentProviderResult[numOperations];  
  5.        for (int i = 0; i < numOperations; i++) {//遍历数据库操作序列  
  6.             results[i] =operations.get(i).apply(this, results, i);//执行数据库操作  
  7.        }  
  8.        return results;//返回结果  
  9.    }  


从上面的代码中,我们找不到和db.beginTransaction()、db.endTransaction()相似的方法,也就是说,这个方法只是进行简单的批处理,并没有保障这些数据库操作的原子性。

好吧。我们稍微动下脑筋,覆写ContentProvider的applyBatch()方法,为其添加事务处理功能。代码如下:

[java]  view plain copy
  1. @Override  
  2. publicContentProviderResult[] applyBatch(ArrayList<ContentProviderOperation>operations)  
  3.             throwsOperationApplicationException{  
  4.           SQLiteDatabasedb = mOpenHelper.getWritableDatabase();  
  5.           db.beginTransaction();//开始事务  
  6.           try{  
  7.                    ContentProviderResult[]results = super.applyBatch(operations);  
  8.                    db.setTransactionSuccessful();//设置事务标记为successful  
  9.                    returnresults;  
  10.           }finally {  
  11.                    db.endTransaction();//结束事务  
  12.           }  
  13. }  


然后,我们该如何使用这个applyBatch()方法呢?applyBatch()的第一个参数实现事务的Provider的authority属性,第二个参数是数据库操作序列,构建数据库操作的对象使用了builder设计模式,下面是一个使用applyBatch()的例子:

[java]  view plain copy
  1. ArrayList<ContentProviderOperation>ops = new ArrayList<ContentProviderOperation>();  
  2. ops.add(ContentProviderOperation.newDelete(Person.CONTENT_URI).build());//添加一个删除Person表的操作  
  3. ops.add(ContentProviderOperation.newInsert(Home.CONTENT_URI).withValues(values).build());//添加一条记录到Home表  
  4. getContentResolver().applyBatch(PROVIDER.AUTHORITY,ops);//处理事务  

总结:

1、sqlite支持事务处理操作

2、对于封装成ContentProvider的sqlite数据库,我们可以通过覆写ContentProvider的applyBatch(Stringauthority, ArrayList<ContentProviderOperation> operations)方法来实现对事务处理的支持

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值