android数据库操作的基础有三个类:SQLiteOpenHelper,SQLiteDatabase和Cursor。其中,SQLiteOpenHelper会建立一个数据库连接,它虽然可以调用多次getWritableDatabase或getReadableDatabase方法,但在彻底关闭db之前,返回的db对象其实是同一个。

    也就是说,如果实例化两个SQLiteOpenHelper,建立两个数据库连接同时操作数据库,会报出数据库已锁定的异常,操作将无效。

    而对于db对象,如果将其彻底关闭,就不能再用来操作数据库,必须重新获取一次。而获取db的过程是比较费时的。



    关于android的db对象,让我困惑的一点是db对象用完后是否要关闭。网上一般有这样几种意见:

    1 db对象开启后就用不关闭。

    2 保持一个单例的db引用,在程序退出前或不再使用数据库时关闭。

    3 每次使用db都要关闭。

4 使用AtomicInteger 计数器来控制db的开启和关闭。

我感觉上面的方案都有其问题:

    1让人感觉就不对头,而且android会不停的报出由于数据库永不关闭造成溢出的警告。    2单例的db问题是,首先无法区分readable和writable,而且很多时候无法确定app退出或不使用数据库的时间点。

    3每次都关闭db,让人怀疑是否会影响性能。

    4用计数器来维持db的引用,增加了代码复杂性。另外,db本身的close方法只是释放了一个引用,只有当所有的引用都被释放后,db才会真正关闭。因此我怀疑计数器是否和db本身的控制机制重复了。



    我想,在没有明显性能影响的情况下,还是每次都关闭db,也能让android日志输出好看一点。



    但即使这样,还是有一个问题,当用ContentProvider返回一个Cursor时,db是不能关闭的,而调用方只能自己关掉Cursor,无法关闭db。不知道ContentProvider是否有一个机制自动关闭db。



    关于Cursor,Cursor使用完必须关闭,否则很容易内存溢出,android也必会报出异常。为了保证Cursor的关闭,可以将Cursor的生命周期和Activity等组件绑定,或者将close方法放在finally代码块中。



    还有一个问题,是Cursor在ContentProvider中跨进程传递机制。我自定义了一个继承MatrixCursor的类,重写了getInt方法,但在调用方处,getInt方法并没有被覆盖。这说明并不是Cursor对象跨进程边界被传递了,具体机制以后有机会可以研究下。

关于android的SQLiteDatabase和Cursor的一些疑问的更多相关文章

  1. Android中SQLite下 Cursor的使用。

    引自博客大神一篇文   地址:  http://blog.sina.com.cn/s/blog_15e2abdd90102wcdu.html rawQuery()方法用于执行select语句.  /* ...

  2. Android——使用SQLiteDatabase操作SQLite数据库

    除了可以使用文件或SharedPreferences存储数据,还可以选择使用SQLite数据库存储数据. 在Android平台上,集成了一个嵌入式关系型数据库-SQLite,SQLite3支持 NUL ...

  3. Android——采用SQLiteDatabase操作SQLite数据库

    除了能够使用文件或SharedPreferences存储数据.还能够选择使用SQLite数据库存储数据. 在Android平台上,集成了一个嵌入式关系型数据库-SQLite,SQLite3支持 NUL ...

  4. Android 开发笔记 “Sqlite Cursor 使用”

    使用过 SQLite 数据库的童鞋对 Cursor 应该不陌生,如果你是搞.net 开发你大可以把Cursor理解成 Ado.net 中的数据集合相当于dataReader.今天特地将它单独拿出来谈, ...

  5. Android 中关于 【Cursor】 类的介绍

    转自(http://www.cnblogs.com/TerryBlog/archive/2010/07/05/1771459.html) 使用过 SQLite 数据库的童鞋对 Cursor 应该不陌生 ...

  6. 单独谈谈 Android Cursor 的使用细节

    使用过 SQLite 数据库对 Cursor 应该不陌生,这里单独拿出来谈一下,加深对Android SQLite中使用 Cursor 的理解. 在你理解和使用 Android Cursor 的时候你 ...

  7. Android中SQLite数据库操作(2)——使用SQLiteDatabase提供的方法操作数据库

    如果开发者对SQL语法不熟,甚至以前从未使用过任何数据库,Android的SQLiteDatabase提供了insert.update.delete或query语句来操作数据库. 一.insert方法 ...

  8. (转) Android开发性能优化简介

    作者:贺小令 随着技术的发展,智能手机硬件配置越来越高,可是它和现在的PC相比,其运算能力,续航能力,存储空间等都还是受到很大的限制,同时用户对手机的体验要求远远高于PC的桌面应用程序.以上理由,足以 ...

  9. Android ORM 框架之 greenDAO 使用心得

    前言 我相信,在平时的开发过程中,大家一定会或多或少地接触到 SQLite.然而在使用它时,我们往往需要做许多额外的工作,像编写 SQL 语句与解析查询结果等.所以,适用于 Android 的ORM  ...

随机推荐

  1. JavaScript split()

    http://www.w3school.com.cn/jsref/jsref_split.asp

  2. C#在foreach循环中修改字典等集合出错的处理

    C#在foreach循环中修改字典等集合出错:System.InvalidOperationException: Collection was modified; enumeration operat ...

  3. Win 7 通过事件管理器查看计算机开机关机时间

    控制面板-管理工具-事件查看器 视图中开机来源:Kernel-General 事件ID:13 关机来源:Kernel-General 事件ID:12

  4. HDU 5904 - LCIS (BestCoder Round #87)

    HDU 5904 - LCIS [ DP ]    BestCoder Round #87 题意: 给定两个序列,求它们的最长公共递增子序列的长度, 并且这个子序列的值是连续的 分析: 状态转移方程式 ...

  5. nyoj 最少步数

    算法:搜索(深度优先搜索) 描述 这有一个迷宫,有0~8行和0~8列: 1,1,1,1,1,1,1,1,1 1,0,0,1,0,0,1,0,1 1,0,0,1,1,0,0,0,1 1,0,1,0,1, ...

  6. encodeURI后台乱码(解决)

    window.location.href = xxxx?a=encodeURI(encodeURI(name)) ; name是中文,页面部分需要编码两次 name = java.net.URLDec ...

  7. javascript 笔试题之删除数组重复元素

    笔试时紧张没写出来,静下心后发现简单的要死. <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" & ...

  8. Symfony2 Doctrine从现有Database生成Entity(转载自http://blog.it985.com/6809.html)

    在我的以前一章Symfony之十分钟入门说了怎样生成数据库,然后设计实体Entity,再同步数据库的表结构,一般我们的顺序都是这样:生成数据库->设计实体Entity->同步数据库表结构. ...

  9. Effective Java2读书笔记-对于所有对象都通用的方法(三)

    第12条:考虑实现Comparable接口 这一条非常简单.就是说,如果类实现了Comparable接口,覆盖comparaTo方法. 就可以使用Arrays.sort(a)对数组a进行排序. 它与e ...

  10. 关于log4net 生成多个文件夹的解决方案。

    解决方案: 在append节点内加入: <param name="lockingModel" type="log4net.Appender.FileAppender ...