Android中SQLiteDatabase操作【附源码】

  像我们做的很多应用程序及网站一样,基本都是对数据库进行增删改查来实现相应的功能。那么Android开发也一样,不过由于在移动客户端应用,所以不会像sql server、mysql那么复杂,Android应用程序支持本地数据库,SQLiteDatabase,通俗的说就是在手机上我们开发的应用程序中创建一个数据库,然后我们可以在手机上对我们的数据进行增删改查,不过这并不是绝对的,像前段时间我们开发一个小组OA,需要多人使用,功能简单,但需要大家连接到一个数据库中进行数据读取操作,所以这种情况下就要考虑到用mysql这样的数据库,最后选择了用php操作后台,然后然会Android进行数据处理,不过对于我们使用的2G网络很多程度上对速度还是有影响的。缓存这一块接触的比较少,计划等到Android这一块学的差不多了再研究其稍底层的一些开发。

  这篇文章主要向大家分享如何操作SQLiteDatabase。 

  当然首先我们要了解SQLiteDatabase,它具有很多优点:

SQLite特性:

    1. 轻量级

    2. 独立性

    3. 隔离性

        SQLite数据库中所有的信息(比如表、视图、触发器等)都包含在一个文件内,方便管理和维护。

    4. 跨平台

    5. 多语言接口

    6. 安全性

        SQLite数据库通过数据库级上的独占性和共享锁来实现独立事务处理。这意味着多个进程可以在同一时间从同一数据库读取数据,但只有一个可以写入数据。在某个进程或现成向数据库执行操作之前,必须获得独占锁定。在发出独占锁定以后,其他的读或写操作将不会再发生。

 

创建和打开数据库:

    openOrCreateDatabase(),自动检测是否存在这个数据库,如果存在则打开,否则创建,创建成功会返回一个SQLiteDatabase对象,否则抛出异常FileNotFoundException:

        mSQLiteDatabase = this.openOrCreateDatabase("abc.db",MODE_PRIVATE,null);

创建表:

    execSQL():

        String Create_Table = "Create table table1...";

        mSQLiteDatabase.execSQL(Create_Table);

 

向表中添加数据:

    insert方法需要把数据打包到ContentValues中,ContentValues其实就是一个Map,Key值是字段名称,Value值是字段的值。通过ContentValues的put方法就可以把数据放到ContentValues对象中,然后插入到表中:

        ContentValue cv = new ContentValues();

        cv.put(table_num,1);

        mSQLiteDatabase.insert(TABLE_NAME,null,cv);

 

从表中删除数据:

    delete():

        mSQLiteDatabase.delete("abc.db","where...",null);

 

修改表数据:

    update():

        ContentValues cv = new ContentValues();

        cv.put(TABLE_NUM,3);

        mSQLiteDatabase.update("table1",cv,"num"+"="+Integer.toString(0),null);

 

  当然,插入、删除和修改操作也可以通过execSQL(sql)方法来实现。

 

关闭数据库:

        mSQLiteDatabase.close();

 

删除指定表:

        mSQLiteDatabase.execSQL("DROP TABLE table1");

 

删除数据库:

        this.deleteDatabase("abc.db");

 

查询表中的某条记录:

        通过Cursor类实现,当使用SQLiteDatabase.query()方法时,会得到一个Cursor对象,Cursor指向的就是每一条数据。它提供了很多有关查询的方法:

            

方法  

说明

move

以当前位置为参考,将Cursor移动到指定的位置,成功返回true

moveToPosition

将Cursor移动的指定的位置,返回boolean

moveToNext

将Cursor向前移动一个位置,返回boolean

moveToLast

将Cursor向后移动一个位置,返回boolean

moveToFirst

将Cursor移动的第一行,返回boolean

isBeforeFirst

返回Cursor是否指向第一项数据之前

isAfterLast

返回Cursor是否指向最后一项数据之后

isClosed

返回Cursor是否关闭

isFirst

返回Cursor是否指向第一项数据

isLast

返回Cursor是否指向最后一项数据

isNull

返回指定位置的值是否为null

getCount

返回总的数据项数

getInt

返回当前行指定索引的数据

 例如:

        Cursor cur = mSQLiteDatabase.rawQuery("select * from table",null);

        if(cur != null)

        {

            if(cur.moveToFirst())

            {

                do{

                    int numColumn = cur.getColumnIndex("num");

                    int num = cur.getInt(numColumn);

                  }while(cur.moveToNext());

            }

        }

     使用SQLiteDatabase数据库后要及时关闭,否则可能会抛出SQLiteException异常。    

 

上面的方法像大部分基础语法书上一样直接执行sql语句的形式,那么在Android中为了简化用户操作以及提高性能,Android系统提供了SQLiteOpenHelper,封装了常用的数据库操作方法。利用它我们可以很轻松的完成对数据库的增删改查。  

 首先我们创建一个DBHelper类继承SQLiteOpenHelper,用它来完成数据库的初始化工作:创建数据库,创建表等操作。

他包含一些借口方法,在下面的注释里已经注释的很详细,就不再罗嗦。

 1 package com.example.core;
 2 
 3 import android.content.Context;
 4 import android.database.sqlite.SQLiteDatabase;
 5 import android.database.sqlite.SQLiteDatabase.CursorFactory;
 6 import android.database.sqlite.SQLiteOpenHelper;
 7 
 8 public class DBHelper extends SQLiteOpenHelper{
 9 
10     public DBHelper(Context context) {
11         //创建数据库名为march_test.db的数据库
12         super(context,"march_test.db",null,1);
13     }
14 
15     /* (non-Javadoc)
16      * 数据库每次被创建时调用
17      * @see android.database.sqlite.SQLiteOpenHelper#onCreate(android.database.sqlite.SQLiteDatabase)
18      */
19     @Override
20     public void onCreate(SQLiteDatabase db) {
21         //创建数据库表
22         String create_sql = "CREATE TABLE student(id integer primary key autoincrement," +
23                 "name varchar(20),age integer not null)";
24         db.execSQL(create_sql);
25     }
26 
27     /* (non-Javadoc)
28      * 版本号发生变化时执行
29      * @see android.database.sqlite.SQLiteOpenHelper#onUpgrade(android.database.sqlite.SQLiteDatabase, int, int)
30      */
31     @Override
32     public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
33         // TODO Auto-generated method stub
34         String alter_sql = "ALTER TABLE student ADD money integer";
35         db.execSQL(alter_sql);
36     }
37     
38 }

  通过实例化这个类,可以创建一个名为march_test.db的数据库。包含数据表student。可以文件系统中看到:

  路径为data/data/包名/databases/数据库名:

    

 

  这种db格式的数据库在这里给大家推荐一个非常好用的工具SQLite Expert Professional,非常好用,在网上也很好找到。他mysql workbench等数据库可视化工具一样给我们提供了可视化数据库操作,软件界面如下:

    

  我们可以把我们创建的表在文件系统中导出来然后放到这里查看。

 

  首先要声明我们要操作的数据类型类:

 1 package com.example.sqllite;
 2 
 3 public class Student{
 4     
 5     private Integer id;
 6     private String name;
 7     private Integer age;
 8     
 9     public Student(Integer id, String name, Integer age) {
10         super();
11         this.id = id;
12         this.name = name;
13         this.age = age;
14     }
15     
16     public Student(String name , Integer age){
17         super();
18         this.name = name;
19         this.age = age;
20     }
21 
22     public Integer getId() {
23         return id;
24     }
25     
26     public void setId(Integer id) {
27         this.id = id;
28     }
29     
30     public String getName() {
31         return name;
32     }
33     
34     public void setName(String name) {
35         this.name = name;
36     }
37     
38     public Integer getAge() {
39         return age;
40     }
41     
42     public void setAge(Integer age) {
43         this.age = age;
44     }
45 
46     @Override
47     public String toString() {
48         return "Student [id=" + id + ", name=" + name + ", age=" + age + "]";
49     }
50     
51 }

 

 

下面要编写对数据库的增删改查类,它继承我们上面创建的SQLiteOpenHelper为基类的DBHelper类:

  1 package com.example.core;
  2 
  3 import java.util.ArrayList;
  4 import java.util.List;
  5 
  6 import android.content.Context;
  7 import android.database.Cursor;
  8 import android.database.sqlite.SQLiteDatabase;
  9 
 10 import com.example.sqllite.Student;
 11 
 12 /**
 13  * @author fanchangfa
 14  *数据库操作类
 15  *增删改查
 16  *获取分页查询数据
 17  *获取表中记录总数
 18  */
 19 public class DbServer{
 20     
 21     private DBHelper dbhelper;
 22     
 23     public DbServer(Context context){
 24         this.dbhelper = new DBHelper(context);        
 25     }
 26     
 27     /**
 28      *  增加信息
 29      * @param student    增加的学生信息
 30      */
 31     public void add(Student student){
 32         SQLiteDatabase db = dbhelper.getWritableDatabase();
 33         db.execSQL("insert into student(name , age) values(?,?)",
 34                 new Object[]{student.getName(),student.getAge()});
 35     }
 36     
 37     /**
 38      * 删除信息
 39      * @param id    要删除的学生id
 40      */
 41     public void delete(Integer id){
 42         SQLiteDatabase db = dbhelper.getWritableDatabase();
 43         db.execSQL("delete from student where id = ?",new Object[]{id});
 44     }
 45     
 46     /**
 47      * 修改指定id的学生信息
 48      * @param stu    包括修改学生的id,以及修改的信息
 49      */
 50     public void alter(Student stu){
 51         SQLiteDatabase db = dbhelper.getWritableDatabase();
 52         db.execSQL("update student set name=?,age=? where id=?",
 53                 new Object[]{stu.getName(),stu.getAge(),stu.getId()});
 54     }
 55     
 56     /**
 57      * 查找信息
 58      * @param id    要查找的学生id
 59      */
 60     public Student find(Integer id){
 61         SQLiteDatabase db = dbhelper.getReadableDatabase();
 62         Cursor cursor = db.rawQuery("select * from student where id = ?",new String[]{id.toString()});
 63         
 64         if(cursor.moveToFirst())    //如果查询结果集中有数据,将游标指向第一条记录
 65         {
 66             int sid = cursor.getInt(cursor.getColumnIndex("id"));
 67             String name = cursor.getString(cursor.getColumnIndex("name"));
 68             int age = cursor.getInt(cursor.getColumnIndex("age"));
 69             
 70             return new Student(sid , name , age);
 71         }
 72         
 73         cursor.close();
 74         
 75         return null;
 76     }
 77     
 78     /**
 79      * 分页查询数据
 80      * @param start    分页开始记录数
 81      * @param end    分页结束记录数
 82      * @return        查询结果集
 83      */
 84     public List<Student> page(int start , int end){
 85         SQLiteDatabase db = dbhelper.getReadableDatabase();
 86         List<Student> page = new ArrayList<Student>();
 87         Cursor cur = db.rawQuery("select id,name,age from student order by id limit ?,?",
 88                 new String[]{String.valueOf(start),String.valueOf(end)});
 89         
 90         while(cur.moveToNext()){
 91             int id = cur.getInt(cur.getColumnIndex("id"));
 92             String name = cur.getString(cur.getColumnIndex("name"));
 93             int age= cur.getInt(cur.getColumnIndex("age"));
 94             page.add(new Student(id,name,age));
 95         }
 96         
 97         cur.close();
 98         
 99         return page;
100     }
101     
102     /**
103      * 返回指定分页数据
104      * @param start
105      * @param end
106      * @return Cursor型数据
107      */
108     public Cursor curpage(int start , int end){
109         SQLiteDatabase db = dbhelper.getReadableDatabase();
110         Cursor cur = db.rawQuery("select id as _id,name,age from student order by id limit ?,?",
111                 new String[]{String.valueOf(start),String.valueOf(end)});
112         
113         cur.moveToFirst();
114         
115         return cur;
116     }
117     
118     /**
119      * 获取表记录总数
120      * @return
121      */
122     public long getCount(){
123         SQLiteDatabase db = dbhelper.getReadableDatabase();
124         
125         Cursor cur = db.rawQuery("select count(*) from student",null);
126         cur.moveToFirst();
127         
128         long count = cur.getLong(0);
129         
130         cur.close();
131         
132         return count;
133     }
134     
135     /**
136      * 执行事务
137      */
138     public void transaction(){
139         SQLiteDatabase db = dbhelper.getWritableDatabase();
140         db.beginTransaction();
141         
142         try{
143             db.execSQL("update student set age = 21 where id =5");
144             db.execSQL("update student set age= 22 where id=6");
145             db.setTransactionSuccessful();
146             //事务默认有commit、rollback,默认为False,即非提交状态,需要设置为commit
147         }
148         finally{
149             db.endTransaction();
150         }
151         
152     }
153 }

 

 

具体操作代码中已经注释完善,可以进行试验。

下面要对其进行测试:

编写测试单元如下:

 1 package com.example.test;
 2 
 3 import java.util.List;
 4 
 5 import com.example.core.DbServer;
 6 import com.example.sqllite.Student;
 7 
 8 import android.test.AndroidTestCase;
 9 import android.util.Log;
10 
11 /**
12  * @author fanchangfa
13  * 数据库操作单元测试
14  * 测试DbServer中数据的增删改查
15  *
16  */
17 public class DbServerTest extends AndroidTestCase{
18     
19     //控制台打印信息标志
20     private static final String TAG = "SQLtest";
21     
22     /**
23      * 添加数据测试
24      */
25     public void addTest(){
26         DbServer dbserver = new DbServer(this.getContext());
27         for(int i = 0 ; i<20 ; i++)
28         {
29             Student stu = new Student("fanchangfa"+i,20);
30             dbserver.add(stu);
31         }
32     }
33     
34     public void deleteTest(){
35         DbServer dbserver = new DbServer(this.getContext());
36         dbserver.delete(2);
37     }
38     
39     public void alterTest(){
40         DbServer dbserver = new DbServer(this.getContext());
41         Student stu = dbserver.find(3);
42         stu.setName("liuzihang");
43         stu.setAge(25);
44         dbserver.alter(stu);
45     }
46     
47     /**
48      * 测试数据库查找
49      * 根据提供id返回记录结果 
50      */
51     public void findTest(){
52         DbServer dbserver = new DbServer(this.getContext());
53         Student stu = dbserver.find(5);
54         Log.i(TAG, stu.toString());
55     }
56     
57     /**
58      * 数据库查找分页测试 
59      */
60     public void findpage(){
61         DbServer dbserver = new DbServer(this.getContext());
62         List<Student> students = dbserver.page(0, 8);
63         
64         for(Student stu : students){
65             Log.i(TAG,stu.toString());
66         }
67         
68     }
69     
70     /**
71      * 执行事务测试
72      */
73     public void transactionTest(){
74         DbServer dbserver = new DbServer(this.getContext());
75         dbserver.transaction();
76     }
77     
78 }

 

经验证,没有问题,由于此文件系统和操作比较麻烦,我将自己写的实例放到这里共大家下载,此实例中包括数据库的操作以及SQLite中事物的使用,以及将在下一篇写的关于ListView显示数据的几种方法,界面虽然很难看,不过这只是demo,希望多多谅解,有问题多多交流。希望这里会是我们拥有共同爱好的程序员们相互交流共同进步的平台,而不是只是为了增加访问量而将文章放在这里。

 

转载于:https://www.cnblogs.com/fanchangfa/archive/2012/08/21/2649710.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/361274.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【BEV感知算法概述——下一代自动驾驶感知算法】

文章目录 BEV感知算法概念BEV感知算法数据集介绍BEV感知算法分类BEV感知算法的优劣小结 BEV感知算法概念 Bird’s-Eye-View&#xff0c;鸟瞰图&#xff08;俯视图&#xff09;。BEV感知算法存在许多的优势。 首先&#xff0c;BEV视图存在遮挡小的优点&#xff0c;由于视觉的透…

博客园CodingLife模板样式优化

博客园CodingLife模板样式优化&#xff0c;小屏和大屏均做了优化&#xff0c;感兴趣的园友可以复制到你的页面定制CSS代码中&#xff0c; 源码地址&#xff1a;http://www.cnblogs.com/blog/customcss/334547.css 更多专业前端知识&#xff0c;请上 【猿2048】www.mk2048.com

WPF控件和布局

WPF控件和布局&#xff0c;根据刘铁猛《深入浅出WPF》书籍讲解内容&#xff0c;主要记录控件和布局的原理&#xff0c;如果有不足的地方&#xff0c;请大牛们键盘下留情--轻喷&#xff01;如果还算有用&#xff0c;请给点动力&#xff0c;支持一把&#xff01; 一、WPF里的控件…

Oracle ——概述 Oracle 5 步调优方法论

http://www.toadworld.com/KNOWLEDGE/KnowledgeXpertforOracle/tabid/648/TopicID/OPS3/Default.aspx 对 Oracle 调优应该采取积极的态度。如果等到用户开始抱怨性能&#xff0c;才调优通常以为时已晚&#xff0c;即便是最有效的调优策略。性能问题确定和处理的时间越晚&#x…

整体服务器与微服务

介绍 刚开始时&#xff0c;由于要求简单&#xff0c;所以应用程序既简单又小。 随着时间要求和需求的增长&#xff0c;我们的应用程序变得越来越大&#xff0c;越来越复杂。 这导致将单片服务器开发和部署为一个单元。 在某种程度上&#xff0c;微服务可以通过简单的应用程序回…

NEXUS S安卓4.0/4.1 【完美】 ROOT教程

原文链接&#xff1a;http://bbs.gfan.com/android-3517082-1-1.html 进行bootloader解锁&#xff08;即使解锁&#xff0c;再上锁&#xff09;&#xff0c;会清除你手机上的【所有】数据&#xff08;包括内部的16G SD 存储&#xff09;&#xff0c;包括但不限于应用、设置、联…

HDU 2647 Reward 拓扑排序

http://acm.hdu.edu.cn/showproblem.php?pid2647 题意&#xff1a; 输入N和M代表N个人和M组数据&#xff0c;M组数据中的A和B代表A的工资要比B的工资高&#xff0c;底薪是&#xff08;888元&#xff09;&#xff0c;问你这个老板至少要付 多少钱给这些员工&#xff0c;A比B工资…

提高Java的锁性能

Plumbr是唯一通过解释应用程序性能数据来自动检测Java性能问题的根本原因的解决方案。 几个月前&#xff0c;我们在Plumbr中引入了锁定线程检测之后&#xff0c;我们开始收到类似于“嘿&#xff0c;太好了&#xff0c;现在我知道是什么导致了性能问题&#xff0c;但是现在应该…

[置顶] UDP协议---心德(1)

UDP协议1.面向无连接 2.速度快 3.不可靠的协议&#xff0c;容易丢包 4.包小于64k DatagramSocket:此类表示用来发送和接收数据报包的套接字 DatagramPacket:数据包 发送端&#xff1a; 1.创建一个udpsocket服务 DatagrameSocket dsnew DatagramSocket(); 2.将数据封装到数据包…

非捕获Lambda的实例

大约一个月前&#xff0c;我在Java 8的lambda表达式框架下总结了Brian Goetz的观点 。 目前&#xff0c;我正在研究有关默认方法的文章&#xff0c;令我惊讶的是&#xff0c;我又回到了Java处理lambda表达式的方式。 这两个功能的交集可能会产生微妙但令人惊讶的效果&#xff0…

SQL Server 查询性能优化——创建索引原则(一)

索引是什么&#xff1f;索引是提高查询性能的一个重要工具&#xff0c;索引就是把查询语句所需要的少量数据添加到索引分页中&#xff0c;这样访问数据时只要访问少数索引的分页就可以。但是索引对于提高查询性能也不是万能的&#xff0c;也不是建立越多的索引就越好。索引建少…

WordPress强制跳转https教程

在互联网火热的今天&#xff0c;安全问题显得越来越重要&#xff0c;为了用户信息安全&#xff0c;很多热门网站都启用了https 有小伙伴就问&#xff1a;我启用了https&#xff0c;为什么访问的时候显示的还是http呢&#xff1f; 其实&#xff0c;有时候并不是因为我们ssl证书…

Java 8的装饰器模式

在最近的一篇文章中&#xff0c;我描述了装饰器模式如何拯救了我的一天。 我给出了一个小代码段&#xff0c;其中包含创建装饰器的最简单方法&#xff0c;但承诺Java 8会有更好的方法。 这里是&#xff1a; 用Java 8装饰 HyperlinkListener listener this::changeHtmlViewBa…

WPF中使用流文档灵活地显示内容

WPF中使用流文档灵活地显示内容 by: Markus Egger form: http://msdn.microsoft.com/msdnmag/issues/07/08/wpf/default.aspx?loczh Windows Presentation Foundation (WPF) 提供了一系列功能。事实上&#xff0c;功能…

canvas图表(4) - 散点图

原文地址&#xff1a;canvas图表(4) - 散点图 今天开始完成散点图&#xff0c;做完这一节&#xff0c;我的canvas图表系列就算是完成了&#xff0c;毕竟平时最频繁用到的就是这几类图表了&#xff1a;柱状&#xff0c;折线&#xff0c;饼图&#xff0c;散点。经过编写canvas图表…

词频统计工程相关

&#xff08;the format of this article is from SkYjoKEr&#xff09; //开始干之前 模块1、WordClass 一个存放单词以及实现相关操作的类&#xff0c;其中单词以二元组<word, freq>的形式存储。 &#xff08;20min&#xff09; 2、WordCounter 完成单词统计&#xff0…

canvas图形编辑器

原文地址&#xff1a;http://jeffzhong.space/2017/11/02/drawboard/   使用canvas进行开发项目&#xff0c;我们离不开各种线段&#xff0c;曲线&#xff0c;图形&#xff0c;但每次都必须用代码一步一步去实现&#xff0c;显得非常麻烦。有没有一种类似于PS&#xff0c;CAD…

Sacrilege –自定义SWT滚动条

SWT是本机OS小部件之上的薄抽象层。 如果您打算将应用程序与OS外观很好地集成在一起&#xff0c;那将是一件非常好的事情。 但是&#xff0c;作为一种折衷方案&#xff0c;这种方法大大限制了样式功能。 特别是&#xff0c;我感觉到本机SW​​T滚动条通常会干扰更精细的视图布…

CSS3盒模型温故

CSS有一种基础设计模式叫盒模型&#xff0c;定义了Web页面中的元素是如何看做盒子来解析的。每一个盒子有不同的展示界面&#xff0c;下面就来介绍盒模型&#xff0c;主要有一下几种盒模型&#xff1a;inline、inline-block、block、table、absolute position、float。浏览器把…

SSL与WildFly 8和Undertow

我一直在研究WildFly 8的一些安全性主题&#xff0c;偶然发现了一些配置文档&#xff0c;这些文档没有很好地记录。 其中之一是新Web子系统Undertow的TLS / SSL配置。 有许多关于较旧的Web子系统的文档&#xff0c;并且确实仍然可以使用&#xff0c;但是这里是使用新方法进行配…