关于SQLite数据库的那些事儿

标签: sqlite 数据库 | 发表时间:2012-02-28 17:36 | 作者:
出处:http://www.iteye.com

 

一、基础知识

在Android平台上,集成了一个嵌入式关系型数据库——SQLite。SQLite第一个Alpha版本诞生于2000年5月。它是遵守ACID的关联式数据库管理系统,包含在一个相对小的C库中,同时是D.RichardHipp建立的公有领域项目。SQLite官方网站: http://www.sqlite.org/;SQLite中文社区:http://www.sqlite.com.cn/。

SQLite3支持NULL、INTEGER、REAL(浮点数字)、TEXT(字符串文本)和BLOB(二进制对象)数据类型,虽然它支持的类型只有五种,但实际上SQLite3也接受varchar(n)、char(n)、decimal(p,s)等数据类型,只不过在运算或保存时会转成对应的五种数据类型。 SQLite最大的特点是你可以把各种类型的数据保存到任何字段中,而不用关心字段声明的数据类型是什么。例如:可以在INTEGER类型的字段中存放字符串,或者在布尔型字段中存放浮点数,或者在字符型字段中存放日期型值。但有一种情况例外:定义为INTEGER PRIMARY KEY的字段只能存储64位整数,当向这种字段保存除整数以外的数据时,将会产生错误。 另外,SQLite在解析CREATE TABLE语句时,会忽略CREATE TABLE语句中跟在字段名后面的数据类型信息,如下面语句中会忽略name字段的类型信息:

CREATE TABLE person (personid integer primary key autoincrement, name varchar(20))即使指定了字段的数据长度,该字段仍然可以保存超过指定长度的内容。

SQLite可以解析大部分标准SQL语句,如:

查询语句:select * from 表名 where 条件子句 group by 分组字句 having ... order by 排序子句

如:select * from person

    select * from person order by id desc

    select name from person group by name having count(*)>1

分页SQL与MySQL类似,下面SQL语句获取5条记录,跳过前面3条记录

select * from person limit 5 offset 3 或者 select * from person limit 3,5

插入语句:insert into 表名(字段列表) values(值列表)。如: insert into person(name, age) values('孙小圣',3)

更新语句:update 表名 set 字段名=值 where 条件子句。如:update person set name=' 孙小圣' where id=10

删除语句:delete from 表名 where 条件子句。如:delete from person where id=10

 

 

二、操作数据库

在Android系统里为我们提供了一个名为SQLiteOpenHelper的抽象类,必须继承它才能使用,它是通过对数据库版本进行管理。为了实现对数据库版本进行管理,SQLiteOpenHelper类提供了两个重要的方法,分别是onCreate(SQLiteDatabase db)和onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion),前者用于初次使用软件时生成数据库表,后者用于升级软件时更新数据库表结构。当调用SQLiteOpenHelper的getWritableDatabase()或者getReadableDatabase()方法获取用于操作数据库的SQLiteDatabase实例的时候,如果数据库不存在,Android系统会自动生成一个数据库,接着调用onCreate()方法,onCreate()方法在初次生成数据库时才会被调用,在onCreate()方法里可以生成数据库表结构及添加一些应用使用到的初始化数据。onUpgrade()方法在数据库的版本发生变化时会被调用,一般在软件升级时才需改变版本号,而数据库的版本是由程序员控制的,假设数据库现在的版本是1,由于业务的变更,修改了数据库表结构,这时候就需要升级软件,升级软件时希望更新用户手机里的数据库表结构,为了实现这一目的,可以把原来的数据库版本设置为2(有同学问设置为3行不行?当然可以,如果你愿意,设置为100也行),并且在onUpgrade()方法里面实现表结构的更新。当软件的版本升级次数比较多,这时在onUpgrade()方法里面可以根据原版号和目标版本号进行判断,然后作出相应的表结构及数据更新。

 

SQLiteOpenHelper类的getWritableDatabase()和getReadableDatabase()方法都可以获取一个用于操作数据库的SQLiteDatabase实例。但getWritableDatabase()方法以读写方式打开数据库,一旦数据库的磁盘空间满了,数据库就只能读而不能写,倘若使用getWritableDatabase()打开数据库就会出错。getReadableDatabase()方法先以读写方式打开数据库,如果数据库的磁盘空间满了,就会打开失败,当打开失败后会继续尝试以只读方式打开数据库。

 

①:编写实现SQLiteOpenHelper类的子类DBOpenHelper

必须显式调用父类的构造方法:super(context, DATABASE_NAME, null, DATABASE_VERSION);

参数2是数据库名称,例如"sxs",可以没有后缀名,也可以有,例如" sxs.db"。

参数3是游标工厂,可以设为null使用默认的游标工厂。

参数4是版本号,例如是"1"。

数据库的位置:/data/data/<package_name>/databases/ sxs.db

 

②:创建数据库表

在onCreate(SQLiteDatabase db)方法中添加如下代码:

db.execSQL("create table person(personid integer primary key autoincrement, name varchar(20))");

 

SQLiteDatabase sQLiteDatabase = DBOpenHelper.getWritableDatabase();//第一次调用该方法就会创建数据库

第一次调用getWritableDatabase()或getReadableDatabase()方法后,SQLiteOpenHelper会缓存当前的SQLiteDatabase实例,SQLiteDatabase实例正常情况下会维持数据库的打开状态,所以你不再需要SQLiteDatabase实例时,请及时调用close()方法释放资源。一旦SQLiteDatabase实例被缓存,多次调用getWritableDatabase()或getReadableDatabase()方法得到的都是同一实例。

 

③:通过SQLiteDatabase进行增删改查。CRUD是指增加(Create)、查询(Retrieve)(重新得到数据)、更新(Update)和删除(Delete)几个单词的首字母简写。

第一种方式:SQLiteDatabase.execSQL("   ")、rawQuery("   ");

execSQL()方法可以执行insert、delete、update和CREATE TABLE之类有更改行为的SQL语句;rawQuery()方法用于执行select语句。rawQuery()方法的第一个参数为select语句,第二个参数为select语句中占位符参数的值,如果select语句没有使用占位符,第二个参数设置为null。rawQuery()方法返回Cursor对象,Cursor是结果集游标,用于对结果集进行随机访问,Cursor与JDBC中的ResultSet作用很相似。使用moveToNext()方法可以将游标从当前行移动到下一行,如果已经移过了结果集的最后一行,返回结果为false,否则为true。另外Cursor还有常用的moveToPrevious()方法(用于将游标从当前行移动到上一行,如果已经移过了结果集的第一行,返回值为false,否则为true )、moveToFirst()方法(用于将游标移动到结果集的第一行,如果结果集为空,返回值为false,否则为true )和moveToLast()方法(用于将游标移动到结果集的最后一行,如果结果集为空,返回值为false,否则为true )。

 

第二种方式:SQLiteDatabase.insert("   ")、delete("   ")、update("   ")和query("   ");

各个字段的数据使用ContentValues进行存放。ContentValues类似于MAP,它提供了存取数据对应的put(String key, Xxx value)和getAsXxx(String key)方法, key为字段名称,value为字段值,Xxx指的是各种常用的数据类型,如:String、Integer等。

 

 

三、关于SQLite数据库中的事务操作

使用SQLiteDatabase的beginTransaction()方法可以开启一个事务,程序执行到endTransaction()方法时会检查事务的标志是否为成功,如果程序执行到endTransaction()之前调用了setTransactionSuccessful()方法设置事务的标志为成功则提交事务,如果没有调用setTransactionSuccessful()方法则回滚事务。使用例子如下:

 

 

SQLiteDatabase db = ....;
db.beginTransaction();//开始事务
try {
    db.execSQL("insert into person(name, age) values(?,?)", new Object[]{"张三", 5});
    db.execSQL("update person set name=? where personid=?", new Object[]{"李四", 1});
    db.setTransactionSuccessful();//调用此方法会在执行到endTransaction()时提交当前事务,如果不调用此方法则回滚事务。
} finally {
    db.endTransaction();//结束事务。由事务的标志决定是提交事务,还是回滚事务
} 
db.close(); 

 上面两条SQL语句在同一个事务中执行。

 

 

 

四、列表显示数据

ListView由多个Item组成,Item可以自定义。每个Item由ImageView(图片)和TextView(文本)组成。

首先定义Item显示界面。

item.xml:

 

	<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
	    android:orientation="horizontal"
	    android:layout_width="fill_parent"
	    android:layout_height="wrap_content"
	    >
		<TextView  
		    android:layout_width="80dip" 
		    android:layout_height="wrap_content" 
		    //android:text="1"
		    android:id="@+id/id"
		    />
		<TextView  
		    android:layout_width="100dip" 
		    android:layout_height="wrap_content" 
		    //android:text="lisi"
		    android:id="@+id/name"
		    />
		<TextView  
		    android:layout_width="fill_parent" 
		    android:layout_height="wrap_content" 
		    //android:text="22"
		    android:id="@+id/age"
		    />
	</LinearLayout>
 

 

其次定义主界面。

main.xml

 

	<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
	    android:orientation="vertical"
	    android:layout_width="fill_parent"
	    android:layout_height="fill_parent"
	    >
	    <Button  
		    android:layout_width="fill_parent" 
		    android:layout_height="wrap_content" 
		    android:text="添加数据"
		    android:id="@+id/insertbutton"
		    />
	    <LinearLayout
		    android:orientation="horizontal"
		    android:layout_width="fill_parent"
		    android:layout_height="wrap_content"
		    >
		    <TextView  
			    android:layout_width="80dip" 
			    android:layout_height="wrap_content" 
			    android:text="ID"
			    />
			<TextView  
			    android:layout_width="100dip" 
			    android:layout_height="wrap_content" 
			    android:text="姓名"
			    />
			<TextView  
			    android:layout_width="fill_parent" 
			    android:layout_height="wrap_content" 
			    android:text="年龄"
			    />
		</LinearLayout>
		<ListView  
		    android:layout_width="fill_parent" 
		    android:layout_height="fill_parent" 
		    android:id="@+id/listView"
		    />
	</LinearLayout>
 

 

然后把数据绑定到ListView中,并为Item指定界面。

 

	//可以使用SimpleAdapter
        List<Person> persons = personService.selectByPage(0, 5);
	//创建Map对象
        List<Map<String, Object>> data = new ArrayList<Map<String,Object>>();
        for(Person person : persons){
        	Map<String, Object> item = new HashMap<String, Object>();
        	item.put("id", person.getId());
        	item.put("name", person.getName());
        	item.put("age", person.getAge());
        	data.add(item);
        }        
        ListView listView = (ListView) this.findViewById(R.id.listView);
        SimpleAdapter adapter = new SimpleAdapter(this, data, R.layout.item, 
        		new String[]{"id", "name", "age"}, new int[]{R.id.id, R.id.name, R.id.age});        
        listView.setAdapter(adapter);

        listView.setOnItemClickListener(new OnItemClickListener(){
		public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
			//parent:当前点击的ListView对象				
			//view:当前点击的条目所绑定的view
			//position:绑定数据的位置
			//id:view在ListView中的id
			ListView aListView = (ListView) parent;
			Map<String, Object> item = (Map<String, Object>) aListView.getItemAtPosition(position);
			Toast.makeText(MainActivity.this, item.get("id").toString(), 1).show();
		}        	
        });


	//也可以使用SimpleCursorAdapter
        /*ListView listView = (ListView) this.findViewById(R.id.listView);
        Cursor cursor = personService.selectCursorByPage(0, 5);
        SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, R.layout.item, cursor,
        		new String[]{"_id", "name", "age"}, new int[]{R.id.id, R.id.name, R.id.age}); 
        listView.setAdapter(adapter);

        listView.setOnItemClickListener(new OnItemClickListener(){
		public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
			//parent:当前点击的ListView对象				
			//view:当前点击的条目所绑定的view
			//position:绑定数据的位置
			//id:view在ListView中的id
			ListView aListView = (ListView) parent;
			Cursor item = (Cursor) aListView.getItemAtPosition(position);
			int personid = item.getInt(item.getColumnIndex("_id"));
			Toast.makeText(MainActivity.this, personid + "", 1).show();
		}        	
        });*/
 

 

当使用SimpleCursorAdapter时,可能会出现以下异常信息:

SimpleCursorAdapter:ERROR/AndroidRuntime(344): Caused by: java.lang.IllegalArgumentException: column '_id' does not exist。

如何解决上述异常?

方法1:创建表时直接定义表的主键列名为"_id"。

方法2:在查询时为表的主键指定别名为"_id"。

 



已有 0 人发表留言,猛击->> 这里<<-参与讨论


ITeye推荐



相关 [sqlite 数据库] 推荐:

将Access转为SQLite数据库

- zhouding - 博客园-首页原创精华区
           SQLite是一个轻量型的数据库,各方面由于Access,对于小型网站来说,非常适合. 我今天决定将系统的数据库由Access转换为SQLite. 在网上搜索了一下,找到了DBManager这个工具,不得不说,这个软件确实好,很强大,在网上找注册码也找到半天,找不到一个.            只要将Access转换为SQLite,然后使用System.Data.SQLite.dll来访问数据库.

SQLite数据库存储引擎设计

- - searchdatabase
  SQLite是一个嵌入式库并且实现了零配置、无服务端和事务功能的SQL数据库引擎. 它在广泛领域内被使用,而且单线程读写性能与MySQL比肩,并且保证ACID性.   SQLite的存储后端是采用Btree实现,多个连接可以并发操作,但是同一时间只允许一个写着存在.   SQLite在硬盘上一个数据库一个文件,每个数据库文件头部保存有这个数据库的元信息,包括版本,大小,Btree根节点位置等等.

关于SQLite数据库的那些事儿

- - ITeye博客
在Android平台上,集成了一个嵌入式关系型数据库——SQLite. SQLite第一个Alpha版本诞生于2000年5月. 它是遵守ACID的关联式数据库管理系统,包含在一个相对小的C库中,同时是D.RichardHipp建立的公有领域项目. SQLite官方网站: http://www.sqlite.org/;SQLite中文社区:http://www.sqlite.com.cn/.

Sqlite数据库分页查询(ListView分页显示数据)

- - CSDN博客推荐文章
今天项目中遇到个问题,之前数据量不算多的时候,ListView显示正常,但是当数据量很大得分时候,进入画面,显示数据比较慢,. 而且不能放在UI线程中去拿数据,用子线程去拿把,画面出来了,但是数据要等很久才会出来,因此,这样给人的体验很不好,算不上好的设计. 因此,查了一下,关于数据库分页ListView分页.

内存数据库FastDB和SQLite性能测评

- - CSDN博客数据库推荐文章
在很多项目中,经常会碰到这样的需求,需要对大量数据进行快速存储、查询、删除等操作,特别是在一些针对诸如运营商、银行等大型企业的应用中,这些需求尤为常见. 比如智能网中的大量在线并发用户的数据管理、软交换平台中的在线信息交互、宽带/3G等数据网中在线用户行为记录等等. 针对这些情形,我们通常需要选择高性能的数据库产品,而且通常需要使用内存数据库,顾名思义,内存数据库指的是所有的数据访问控制都在内存中进行,这是与磁盘数据库相对而言的,磁盘数据库虽然也有一定的缓存机制,但都不能避免从外设到内存的交换,而这种交换过程对性能的损耗是致命的,目前主流数据库如SYBASE、ORACLE等都有这种缓存机制,如将特定表绑定一定的缓存,从而在一定程度上改善数据吞吐性能.

Android SQLite数据库版本升级原理解析

- - CSDN博客推荐文章
Android使用SQLite数据库保存数据,那数据库版本升级是怎么回事呢,这里说一下. 安装v1.0,假设v1.0版本只有一个account表,这时走继承SQLiteOpenHelper的onCreate,不走onUpgrade. 1、v1.0(直接安装v1.0). 1、v1.0   -->  v2.0              不走onCreate,走onUpgrade.

Android学习笔记(六)SQLite

- - 博客园_首页
SQLite是一个极轻量型的数据库. 它在提供了和大型数据库相当的功能,还具有轻便、跨平台等优点,SQLite使用非常方便,并不需要我们像常规数据库(SQLServer,Mysql等)那样进行安装,在Android的JDK中,其实是已经包含了SQLite这个数据库的核心. 当然我们必须要强调一点,SQLite并不是只针对Android的,其实它还可以用在别的很多地方.

SQLite的原子提交原理

- way - chensheng.net
本文源自:http://www.sqlite.org/atomiccommit.html,2007/11/28的版本. 本人正在做一个项目,在项目中定义了自己的文件格式,为了做到停电或程序崩溃不损坏这些文件原有的数据,故针对操作的原子性做一些思考,后来看到sqlite的这篇文章,与自己的实现方式作了一些对比.

sqlite插入速度优化方案

- - CSDN博客移动开发推荐文章
先贴上SQLiteOpenHelper的代码,我们创建测试的表是user,它有四个字段id,name,age,height,remark,测试用的小米2,默认条数为100行. // 必须要有这一个构造方法. // 当数据库第一次创建的时候被调用,. // 当调用getReadableDatabase ()或getWritableDatabase 的时候.

Android数据的四种存储方式SharePreference、SQLite、Content Provider和File (二) —— SQLite

- stony - 博客园-首页原创精华区
  SQLite是一种转为嵌入式设备设计的轻型数据库,其只有五种数据类型,分别是:.     INTEGER: 整数.   在SQLite中,并没有专门设计BOOLEAN和DATE类型,因为BOOLEAN型可以用INTEGER的0和1代替true和false,而DATE类型则可以拥有特定格式的TEXT、REAL和INTEGER的值来代替显示,为了能方便的操作DATE类型,SQLite提供了一组函数,详见:http://www.sqlite.org/lang_datefunc.html.