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,一经查实,立即删除!

相关文章

移动设备HTML5页面布局

在HTML5标准添加的新元素中&#xff0c;用于常见页面结 构的包括header footer footer nav aside aside article section hgroup 。 下面简单介绍一下这个元素&#xff1a; 1.header header>元素定义文档的页面组合&#xff0c;通 常是一些引导和导航信息&#xff0c;标签…

Hazelcast入门指南第5部分

这是我撰写的有关Hazelcast的一系列文章的延续。 我强烈建议您阅读其他内容&#xff1a; 第1 部分 &#xff0c; 第2 部分 &#xff0c; 第3 部分和第4部分 。 让人一见倾心的东西&#xff1f; 这篇文章中将没有Hazelcast专用代码。 让我重复一遍。 这篇文章中将没有Hazelcast…

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

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

jquery鼠标事件

click()   为点击事件绑定一个事件处理函数&#xff0c;或者触发元素点击事件。   .click( handler(eventObject) )     handler(eventObject)       每次事件触发时候执行的函数。   .click([eventData],handler(eventObject))     eventData      …

[LaunchPad] 超声波测试,数码管显示

1 #include "io430g2553.h"2 3 unsigned char Disp_Tab[] {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40}; //段码控制 4 unsigned char dispbit[8]{0xfe,0xfd,0xfb,0xf7,0xef,0xdF,0xbF,0x7F}; //位选控制 5 6 unsigned char LedOut[4];7 8 …

Spring的依赖注入陷阱

Spring框架中有三种注入变量&#xff1a; 基于二传手的注射 基于构造函数的注入 基于现场的注入 这些机制中的每一种都有优点和缺点&#xff0c;并且不仅只有一种正确的方法。 例如现场注入&#xff1a; Autowired private FooBean fooBean;在生产代码中使用它通常不是最好…

jquery 操作css 选择器

.addClass()   为每个匹配的元素添加指定的样式类名   .addClass(className)     className 为每个匹配元素所有增加的一个或多个样式名   .addClass(function(index,currentClass))     函数返回一个或者多个用空格隔开&#xff0c; index 表示参数匹配中的索引…

vim设置

一、基本编辑功能1、复制、剪切和粘贴复制特定的某一段&#xff1a;把光标移到要复制的文本的头部&#xff0c;按下“v”&#xff0c;往后移动光标&#xff0c;光标所过之处的字符>都会高亮&#xff0c;移到欲复制文本的尾部后&#xff0c;按下“y”&#xff0c;高亮文本全部…

jquery表单属性筛选元素

$(":button") 选择所有按钮元素类型为按钮的元素。 等于$(input[type"button"]) $(":checkbox") 选择所有类型为复选框的元素。 等于$(input[type"checkbox"]) $(":checked") 选择所有勾选的元素。 $( "input:checked&…

Hamcrest Matchers的高级创建

介绍 上一次 &#xff0c;我讨论了Hamcrest Matcher是什么&#xff0c;如何使用以及如何制作。 在本文中&#xff0c;我将解释创建Hamcrest Matchers的更多高级步骤。 首先&#xff0c;我将分享如何使您的匹配器更易于类型安全&#xff0c;然后介绍无状态匹配器的一些技术&…

嵌入式成长轨迹37 【Zigbee项目】【CC2430基础实验】【自动闪烁】

最为简单的代码&#xff0c;只用到一个寄存器P1DIR。因为点亮的led灯&#xff08;p1.0和p1.1&#xff09;的管脚是p1的&#xff0c;要输出就得将这两个管脚设置为输出管脚。该寄存器用0~8对应1.0~1.8管脚。 1 //main.c2 #include <ioCC2430.h>3 4 #define uint unsigned …

博客园CodingLife模板样式优化

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

如何允许用户自定义UI

理念 利用JavafX / FXML的声明性设计模式&#xff0c;并允许用户仅通过使用例如SceneBuilder打开某个视图即可重新定制布局或添加新控件&#xff0c;甚至根据用户需要更改样式&#xff0c;从而无需任何编码即可自定义某个视图。 FXML文件 CSS基本上可以放置在通过URL可以到达的…

WPF控件和布局

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

css背景图片定位

背景图默认平铺&#xff1a; background-repeat: no-repeat;/*不平铺*//*repeat-x;沿X轴平铺*//*repeat-x;沿Y轴平铺*/ 背景图片定位&#xff1a; background-position: 100px 30px;/* X轴 Y轴 *//* 20% 20%; 百分比的方式 *//* left|center|right top|center|bottom */ 背景中…

使用GZIP和压缩数据

抽象 我们都知道用zip或gzip压缩文件的含义。 但是在Java中使用压缩文件并不像您想的那样简单明了&#xff0c;尤其是当您不是直接使用文件而是压缩流数据时。 我们会去&#xff1a; 如何将字符串转换为压缩/压缩字节数组&#xff0c;反之亦然 创建用于读取和写入文件的实用程…

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

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

js,jq.事件代理(事件委托)复习。

<ul id "lists"><li>列表1</li><li>列表2</li><li>列表3</li><li>列表4</li><li>列表5</li><li>列表6</li></ul>js委托&#xff1a; var lists document.getElementById(&qu…

整体服务器与微服务

介绍 刚开始时&#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;包括但不限于应用、设置、联…