Android系统存储数据的第四种方式是SQLite Database,即数据库存储。

  该方式(数据库存储)的特点是存储私有数据,且数据是具有数据结构的,存储位置/data/data/<包名>/databases目录下。

  数据库存储有两个核心类:

SQLiteOpenHelper类:  数据库管理类

SQLiteDatabase类:   数据库操作类

  

  SQLiteOpenHelper查阅官方文档知:

  

android本地数据存储 安卓本地存储数据库_数据

从上可知:SQLiteOpenHelper是一个帮助类,用来管理数据库的创建和数据库的版本,一般定义它的子类,实现它的两个方法onCreate,onUpdate。

  eg:

    

android本地数据存储 安卓本地存储数据库_数据库_02

android本地数据存储 安卓本地存储数据库_android本地数据存储_03

1 public class DBHelper extends SQLiteOpenHelper {
 2 
 3     public DBHelper(Context context)
 4     {
 5         //第一个参数是Context对象
 6         //第二个参数是数据库的名字
 7         //第三个参数是管理游标的工厂类对象
 8         //第四个参数是当前数据库的版本
 9         super(context, "user.db", null, 1);
10     }
11 
12     //创建数据库时执行该方法
13     //表中字段必须含有一个  _id
14     @Override
15     public void onCreate(SQLiteDatabase db) {
16 
17         db.execSQL("create table t_user(_id integer primary key,name text,age int,tel text)");
18 
19     }
20 
21     //数据库版本升级时执行该方法
22     @Override
23     public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
24 
25         if(newVersion>oldVersion)
26         {
27             db.execSQL("drop table if exists t_user");
28         }
29     }
30 
31 }

SQLiteOpenHelper子类的创建

  

  SQLiteOpenHelper类的构造方法:

  

android本地数据存储 安卓本地存储数据库_数据_04

  这里我们一般使用第一个有四个参数的构造方法,其参数含有分别为:

    

android本地数据存储 安卓本地存储数据库_android本地数据存储_05

    Context context:  context对象

    String name:    创建的数据库名字

    SQLiteDatabase.CursorFactory factory:  管理数据库中游标的工厂类对象,一般设为null

    int version:    数据库的版本号,用于数据库版本的升级或降级

 

  SQLiteOpenHelper类的公共方法:

  

android本地数据存储 安卓本地存储数据库_SQL_06

    其中我们较长使用的是:

 getReadableDatabase();  获取一个可读(不包含可写)的数据库对象,返回的是一个SQLiteDatabase对象

       getWriteableDatabase();  获取一个可写(包含可读)的数据库对象,返回的是一个SQLiteDatabase对象

 

 

  SQLiteDatabase查阅官方文档知:

    

android本地数据存储 安卓本地存储数据库_数据库_07

根据以上可知SQLiteDatabase是一个实际操作数据库的类,可执行数据库的增删改查操作。

 

    获取SQLiteData对象,我们一般使用上面所介绍的方法:getReadableDatabase()和getWriteableDatabase()。此外我们也可以使用SQLiteDatabase的静态方法打开一个指定的数据库。

使用SQLiteDatabase的静态方法打开数据库: 

      

android本地数据存储 安卓本地存储数据库_android本地数据存储_08

    eg:

1 // 被操作的数据库文件的路径
2     public static final String DB_PATH = Environment
3             .getExternalStorageDirectory() + "/cache/db/gp.db";
4 
5         SQLiteDatabase  db = SQLiteDatabase.openDatabase(DB_PATH, null, SQLiteDatabase.OPEN_READWRITE);

 

    注释:这里打开的数据库文件位于手机的扩展卡上,此处我们也可以知道数据库存储位置也可以位于扩展卡,SQLiteDatabase可以操作扩展卡上的数据库文件。

 

SQLiteDatabase常用操作:

      

android本地数据存储 安卓本地存储数据库_数据存储_09

 

 

数据库存储的操作中,最重要的是查询操作,对于数据库的查询,会返回一个Cursor对象,通过Cursor对象,我们可以获取查询的结果。

   Cursor即游标,是一个与数据库操作紧密相关的类,查阅官方文档知:

      

android本地数据存储 安卓本地存储数据库_数据库_10

Cursor的常用方法:

      moveToNext();

     getColumnCount();

     getColumnNames();  返回一个字符串数组

     getColumnName();

     getColumnIndex();

     getString(),getInt(),...

 

设想这样一种情景:通过数据库查询,获取结果,然后在界面ListView中显示出来。显然我们是通过适配器来适配数据,对于Cursor的使用有如下两种方式:

   方式一:可以使用一般的适配器,从Cursor中获取数据,初始化数据源。

1 private void initListView(){
 2         ListView listView = (ListView) findViewById(R.id.listView);
 3 
 4         List<Map<String,Object>> data = new ArrayList<Map<String,Object>>();
 5 
 6         SQLiteDatabase db = dbHelper.getReadableDatabase();
 7         Cursor cursor = db.rawQuery("select * from t_user",null);
 8         while(cursor.moveToNext()){
 9             Map<String,Object> map = new HashMap<String,Object>();
10             map.put("_id",cursor.getInt(cursor.getColumnIndex("_id")));
11             map.put("name",cursor.getString(cursor.getColumnIndex("name")));
12             map.put("age",cursor.getInt(cursor.getColumnIndex("age")));
13             map.put("tel",cursor.getInt(cursor.getColumnIndex("tel")));
14             data.add(map);
15         }
16 
17         SimpleAdapter adapter =
18                 new SimpleAdapter(this,data,R.layout.item_user,
19                         new String[]{"_id","name","age","tel"},
20                         new int[]{R.id.text_id,R.id.text_name,R.id.text_age,R.id.text_tel});
21 
22         listView.setAdapter(adapter);
23     }

 

     注释:这里dbHelper是一个DatabaseOpenHelper子类的对象,已经提前创建,该处使用的是SimpleAdapter适配器。

 

  方式二:使用与Cursor相对应的适配器SimpleCursorAdapter。

1  private void initListView(){
 2         String[] columns={"_id","name","age","tel"};
 3         int[] to ={R.id.text_id,R.id.text_name,R.id.text_age,R.id.text_tel};
 4 
 5         ListView listView = (ListView) findViewById(R.id.listView);
 6         Cursor cursor = null;
 7         SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, R.layout.item_user,cursor,
 8                 columns,
 9                 to,
10                 SimpleCursorAdapter.FLAG_REGISTER_CONTENT_OBSERVER);
11 
12         listView.setAdapter(adapter);
13 
14 
15         //加载数据
16         SQLiteDatabase db = dbHelper.getReadableDatabase();
17         cursor = db.query("t_user", columns, null, null, null, null, null);
18 
19         //游标中的数据变化了,需要切换适配器的数据源
20         adapter.swapCursor(cursor);
21 
22     }

 

   这里将相应的代码分别全部集中于同一个方法,看起来有些繁琐,当然使用的时候可以从实际情况出发,以上仅为了作为例子的方便使用才这么做。

   对比这两种方式,可知:对于数据库的操作,更多的时候会返回一个Cursor对象,当我们为了在AdapterView中显示数据库中的数据,我们可以使用SimpleCursorAdapter这个类,它可以将AdapterView和Cursor(数据源)直接沟通起来.

   补上数据库添加数据的部分:

1 private void initData() {
 2         SQLiteDatabase db = dbHelper.getWritableDatabase();
 3 
 4         ContentValues values = null;
 5 
 6         for(int i=0;i<20;i++){
 7             values = new ContentValues();
 8             values.put("_id", i);
 9             values.put("name", "Tom "+ i);
10             values.put("age", 18 + i);
11             values.put("tel","2323"+i);
12 
13             db.insert("t_user", null, values);
14         }
15         db.close();
16     }

 

    数据库文件位置:

      

android本地数据存储 安卓本地存储数据库_数据库_11

  采用两种方式呈现的结果相同:

   

android本地数据存储 安卓本地存储数据库_数据存储_12

   关于Cursor和SimpleCursorAdapter的详细内容,请查询相关文档或参见后续文章。

 

   未完,待续。