Android面试之SQLite数据库
更新时间:2024-01-26 00:12:01 阅读量: 教育文库 文档下载
- android 面试推荐度:
- 相关推荐
Android面试之SQLite数据库
鹭岛厦门是个很美丽的海滨城市,给我的感觉很舒适和悠闲,据说政府对到那工作的高新技术人才第一年有10万元的奖励,因为这个原因我很有兴趣的参加了一个厦门公司的面试。他们主要是研发VOIP方面的技术,对手机应用的性能优化和音频算法有较高的要求。
他们招人的薪资半年内涨了30万元,都一直都没有招到合适的人。为什么呢?因为他们要求技术好的同时,英语也要好。什么才叫好呢?就是可以用流利的英语和国外的团队无障碍交流。
这就为难很多程序员了。这里就不再讲英语的励志故事了,我们回到技术面试上。厦门这家公司对技术的要求还是比较高,问了很多对Android机制的理解问题,为什么面试官很在意对机制的理解呢?因为实际的项目中很多性能问题都是由于缺乏对Android运行机制的正确理解的程序员引发的。除了机制问题,现在印象比较深的就是关于SQLite数据库操作的性能优化问题。 面试题:如何对SQLite数据库中进行大量的数据插入?
Android系统内置了SQLite数据库,并且提供了一整套的API用于对数据库进行增删改查操作。SQLite是一个轻量的、跨平台的、开源的数据库引擎。SQLite每个数据库都是以单个文件(.db)的形式存在,这些数据都是以B-Tree的数据结构形式存储在磁盘上。
使用SQLiteDatabase的insert,delete等方法或者execSQL方法默认都开启了事务,如果操作的顺利完成才会更新.db数据库。事务的实现是依赖于名为rollback journal文件,借助这个临时文件来完成原子操作和回滚功能。 大家可以在/data/data//databases/目录下看到一个和数据库同名的.db-journal文件。
SQLite想要执行操作,需要将程序中的SQL语句编译成对应的
SQLiteStatement,比如\,每执行一次都需要将这个String类型的SQL语句转换成SQLiteStatement。如下insert的操作最终都是将ContentValues转成SQLiteStatementi:
public long insertWithOnConflict(String table, String nullColumnHack,
ContentValues initialValues, int conflictAlgorithm) { // 省略部份代码
SQLiteStatement statement = new SQLiteStatement(this, sql.toString(), bindArgs); try {
return statement.executeInsert(); } finally { statement.close(); }
} finally {
releaseReference(); } }
对于批量处理插入或者更新的操作,我们可以重用SQLiteStatement,使用SQLiteDatabase的beginTransaction()方法开启一个事务,样例如下: try {
sqLiteDatabase.beginTransaction();
SQLiteStatement stat = sqLiteDatabase.compileStatement(insertSQL); // 插入10000次
for (int i = 0; i < 10000; i++) {
stat.bindLong(1, 123456); stat.bindString(2, \ stat.executeInsert(); }
sqLiteDatabase.setTransactionSuccessful(); }
catch (SQLException e) {
e.printStackTrace(); } finally { // 结束
sqLiteDatabase.endTransaction(); sqLiteDatabase.close(); }
我在华为Nexus 6P上对常见的几种做法做了一下测试。 直接使用SQL语句进行插入 直接使用SQL语句插入,添加事务 使用ContentValues方式,添加事务 使用SQLiteStatement方式,添加事务 结果如下图:
从数据上看,第四种方式使用SQLiteStatement最快,不过只要添加了事务(或者说只需要一个事务,不是每条插入都使用事务),后三种方式的差别并不大。所以针过这个题目的插入的优化可以通过“SQLiteStatement+事务”的方式显著提高效率。
查询方面的优化一般可以通过建立索引。建立索引会对插入和更新的操作性能产生影响,使用索引需要考虑实际情况进行利弊权衡,对于查询操作量级较大,业务对要求查询要求较高的,还是推荐使用索引。所以这会有一个取舍问题,看你的项目是查询频繁还是插入和修改频繁。当然还有一些小的优化细节,如果面试官问到也可以说几点(如limit)。 线程问题
SQLite的同步锁精确到数据库级,粒度比较大,不像别的数据库有表锁,行锁。同一个时间只允许一个连接进行写入操作。
如果有大量的数据处理,那么肯定不合适于在UI线程去操作,这时就要考虑多线程的问题了。我们如果开一个工作线程去操作SQLite数据库,如批量地插入可能需要30秒钟,而这个时间UI线程也要从数据库读取一下数据展示给用户,那么这个时候UI线程能读取到这个数据库吗?大家可以思考一下这个问题。 我们常常在多线程中只使用一个SQLiteDatabase引用,在用
SQLiteDataBase.close()的时需要注意调是否还有别的线程在使用这个实例。如果一个线程操作完成后就直接close了,别一个正在使用这个数据库的线程就会异常。所以有些人会直接把SQLiteDatabase的实例放在Application中,让它
们的生命周期一致。也有的做法是写一个计数器,当计数器为0时才真正关闭数据库。
使用ORM的问题
目前网上有很多开源的ORM(对象关系数据映射)框架,如greenDAO、ormlite等等。在使用这些框架有必要很了解一下它们的利弊,特别是一些使用反射的框架,对性能的影响会比较大。有些框架在多线程同步方面也会产生一些问题,所以使用时要有所顾虑。
Realm 是最近兴起的一个专注于移动设备数据库的库,其核心是使用C++编写,号称很多时候数据的存取速度比SQLite要快很多。不过在我的一些项目中,发现它读取并不比SQLite快。 小结
在实践中我们总结出一条守则:“不要用Helloworld来测试自己的框架(或代码),要测就要用真实的数据和环境。”
特别是针对数据库方面,如果只用几条简单的数据进行测式,那么你会很容易傲娇和满足,而忽视了很多问题。没有经过真实数据(或大量数据)测试之前,不要对自己的代码太过自信。
正在阅读:
Android面试之SQLite数据库01-26
株洲景炎学校2012年初新招生综合测评 科学素养卷(含答案)01-30
水吸收氨气过程填料吸收塔的设计04-05
关于距离的作文500字07-02
爱因斯坦的名言警句摘抄11-20
文明的距离作文500字06-20
高中班主任个人XX年度工作总结05-09
三相用电设备组计算负荷04-26
中专膏火一年几多钱 收费规范是什么03-30
- exercise2
- 铅锌矿详查地质设计 - 图文
- 厨余垃圾、餐厨垃圾堆肥系统设计方案
- 陈明珠开题报告
- 化工原理精选例题
- 政府形象宣传册营销案例
- 小学一至三年级语文阅读专项练习题
- 2014.民诉 期末考试 复习题
- 巅峰智业 - 做好顶层设计对建设城市的重要意义
- (三起)冀教版三年级英语上册Unit4 Lesson24练习题及答案
- 2017年实心轮胎现状及发展趋势分析(目录)
- 基于GIS的农用地定级技术研究定稿
- 2017-2022年中国医疗保健市场调查与市场前景预测报告(目录) - 图文
- 作业
- OFDM技术仿真(MATLAB代码) - 图文
- Android工程师笔试题及答案
- 生命密码联合密码
- 空间地上权若干法律问题探究
- 江苏学业水平测试《机械基础》模拟试题
- 选课走班实施方案
- 面试
- Android
- 数据库
- SQLite
- 自考02379《计算机网络管理》串讲资料
- 工程现场签证管理及标准模板
- 部编人教版三年级下册9.古诗三首教案-清明
- IPRAN故障排查
- 目标市场STP分析
- 收割机行走系统
- 食品考试
- 公共经济学课后复习思考题
- 《面向对象程序设计》实验册
- 四年级藏文期中试卷
- 井控坐岗员读本
- 9第九章教育科学的调查研究法
- 道教入宅拜家堂谢土神科仪1
- 法律常识500题(打印版)
- 2014版学海导航数学(文)总复习(第1轮)同步测控 第53讲 两直线的位置关系与对称问题 Word版含答案
- 2010年内蒙古赤峰市中考物理试题(word版及答案)
- 医院安全生产责任制
- 第五单元历经风雨才见彩虹复习课
- 深圳市建设局公布的停工窝工费用补偿办法的规定-推荐word版(10页)
- 总承包管理方案(最终版) - 图文