1.Sql基本命令

1.1.创建表

  表是有行和列组成的,列称为字段,行称为记录。

  使用CREATE命令来创建表:

  

1   CREATE TABLE tab_student (studentId INTEGER PRIMARY KEY AUTOINCREMENT, 
2                             studentName VARCHAR(20), 
3                             studentAge INTEGER);

 

 

1.2.插入记录(行)

  使用INSERT命令可以一次插入一条记录,INSERT命令的一般格式为:

  

INSERT INTO tab_student (studentId, studentName, studentAge) VALUES (1, “jack”, 23);

 

1.3.更新记录(行)

  使用UPDATE命令可以更新表中的记录,该命令可以修改一个表中一行或者多行中的一个或多个字段。UPDATE命令的一般格式为:

  

UPDATE tab_student SET studentName=”tom”, studentAge=”25” WHERE studentId=1;

 

 

1.4.删除记录(行)

  使用DELETE命令可以删除表中的记录,DELETE命令的一般格式为:

DELETE FROM tab_student WHERE studentId=1;

 

1.5.查询记录(行)

  SELECT命令是查询数据库的唯一命令。SELECT命令也是SQL命令中最大、最复杂的命令。

 

  SELECT命令的通用形式如下:

  SELECT [distinct] heading 

  FROM tables 

  WHERE predicate 

  GROUP BY columns 

  HAVING predicate 

  ORDER BY columns 

  LIMIT count,offset;

  其中,每个关键字(如FROM、WHERE、HAVING等)都是一个单独的子句,每个子句由关键字和跟随的参数构成。GROUP BY和HAVING一起工作可以对GROUP BY进行约束。ORDER BY使记录集在返回之前按一个或多个字段的值进行排序,可以指定排序方式为ASC(默认的升序)或DESC(降序)。此外,还可以使用LIMIT限定结果集的大小和范围,count指定返回记录的最大数量,offset指定偏移的记录数。

  在上述的SELECT命令通用形式中,除了SELECT之外,所有的子句都是可选的。目前最常用的SELECT命令由三个子句组成:SELECT、FROM、WHERE,其基本语法形式如下:

  SELECT heading FROM tables WHERE predicate;

  比如,要查询刚才插入的记录,便可以使用如下的语句完成:

  

SELECT studentId, studentName, studentAge FROM tab_student WHERE studentId=1;

 

2.数据库操作辅助类SQLiteOpenHelper

 

Android提供了一个重要的类SQLiteOpenHelper,用于辅助用户对SQLite数据库进行操作。

 

  SQLiteOpenHelper的构造函数原型如下:

 

  public SQLiteOpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version);

 

  其中,参数context表示应用程序运行的环境,包含应用程序所需的共享资源。参数name表示Android的数据库名字。参数factory是SQLiteDatabase.CursorFactory类对象,用于存储查询Android SQLite数据库的结果集。参数version表示应用程序所用的数据库的版本,该版本并非SQLite的真正版本,而是指定应用程序中的SQLite数据库的版本,当该版本号发生变化时,将会触发SQLiteOpenHelper类中的onUpgrade()或onDowngrade()方法。

SQLiteOpenHelper类的所有方法如图1所示。

 

  

 

 其中,close()方法用于关闭SQLiteOpenHelper对象中的SQLite数据库;getReadableDatabase()方法和getWriteableDatabase()方法类似,getReadableDatabase()方法以只读状态打开SQLiteOpenHelper对象中指定的SQLite数据库,任何想要修改数据库的操作都是不允许的;getWriteableDatabase()方法也是打开数据库,但是允许数据库正常的读/写操作;在一个不存在的数据库上调用任何方法时,都会隐式的调用SQLiteOpenHelper对象的onCreate()方法;当应用程序第一次访问数据库时,则会调用onOpen()方法,但是,如果版本号发生了变化的话,则会调用onUpgrade()或onDowngrade()方法。

 

3.数据库类SQLiteDatabase

SQLiteDatabase类用来完成对数据库的操作任务,比如表的选择、插入、更新和删除语句等。

 

  SQLiteDatabase类中常用的用于执行SQL语句的方法有以下一些。

 

  (1)execSQL()方法:

 

  public void execSQL (String sql);

 

  public void execSQL (String sql, Object[] bindArgs);

 

  (2)query()方法:

 

  public Cursor query (String table, String[] columns, String selection, String[] selectionArgs, String  groupBy, String  having,String orderBy, String limit);

 

  public Cursor query (boolean distinct, String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit, CancellationSignal cancellationSignal);

 

  public Cursor query (String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String  having,String orderBy);

 

  public Cursor query (boolean distinct, String table, String[] columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit);

 

 

  (3)queryWithFactory()方法:

          public Cursor queryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, boolean distinct, String table, String[]columns, String selection, String[] selectionArgs, String groupBy,  String having, String  orderBy, String  limit,CancellationSignal cancellationSignal);

  public Cursor queryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, boolean distinct, String table, String[]columns, String selection, String[] selectionArgs, String groupBy, String having, String orderBy, String limit);

  (4)rawQuery()方法:

  public Cursor rawQuery (String sql, String[] selectionArgs, CancellationSignal cancellationSignal);

  public Cursor rawQuery (String sql, String[] selectionArgs);

  (5)rawQueryWithFactory()方法:

  public Cursor rawQueryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, String sql, String[]  selectionArgs,String editTable);

  public Cursor rawQueryWithFactory (SQLiteDatabase.CursorFactory cursorFactory, String sql, String[]  selectionArgs,String editTable, CancellationSignal cancellationSignal);

  其中,execSQL()方法都有一个参数sql,这个参数是一个SQL语句。第二个参数bindArgs接收一个数组,数组中的每个成员捆绑了一个查询。execSQL()方法用于运行那些没有返回值的查询语句,比如创建、插入、更新和修改表。

  query()方法和queryWithFactory()方法是在数据库中运行一些轻量级的单查询语句,参数包括table、columns、groupBy、having、orderBy、limit等SQL语句关键字。这些方法允许将SQL语句传递给相关方法,而不必直接使用SQL语句。

  rawQuery()方法和rawQueryWithFactory()方法也都有一个参数sql,用于执行SQL查询语句,返回值是Cursor对象。这两个方法都有一个版本能够接收一个字符串数组selectionArgs作为参数,通过这个参数,SQLiteDatabase对象将把捆绑的SQL语句中的问号(?)用这个数组中的值代替,并按照一一对应的位置关系进行取代。

  SQLiteDatabase类提供了大约50个方法,除此之外还有一些用于打开数据库的方法(如openDatabase()、openOrCreateDatabase()等),用于管理SQLite事务的方法(如beginTransaction()、endTransaction()等),用于测试数据库是否被锁住的方法(如isDbLockedByCurrentThread()、isDbLockedByOtherThread()等),以及获取数据库基本信息的方法(如getMaximumSiza()、getVersion()等)。这里就不一一介绍了,具体可以参阅SQLiteDatabase类的API帮助文档

4.游标类Cursor

在Android中,查询数据是通过Cursor类来实现的,当我们使用SQLiteDatabase.query()或SQLiteDatabase.rawQuery()方法时,会得到一个Cursor对象,Cursor指向的就是每一条记录,它提供了很多有关查询的方法,如图2所示。

 5.封装接口

有了以上的基础,我们便可以按照MVC的架构,封装一个接口层,在该接口层中实现对SQLite数据库的具体操作。

  以下分别以添加数据、更新数据、查询数据为例讲解其具体的实现方法。在实现这些方法之前,我们首先需要创建一张表。这里我创建了一个名为MySQLiteOpenHelper的类,让它继承自SQLiteOpenHelper类,并实现了SQLiteOpenHelper类的onCreate()方法,在该方法里实现创建一张表的操作,具体源代码如下:

1     /*
2      * Function  :    创建表
3      * Author    :    博客园-依旧淡然
4      */
5     public void onCreate(SQLiteDatabase db) {
6         db.execSQL("CREATE TABLE tab_student (studentId INTEGER PRIMARY KEY AUTOINCREMENT, " + 
                                               "studentName VARCHER(20), " +               
                                               "studentAge INTEGER)");
7     }

通过以上的代码,我们创建了一张名为“tab_student”的表,并在该表中创建了三个字段,分别为:studentId、studentName和studentAge。并且指定了studentId字段作为该表的主键。

 

5.1添加数据

添加数据可以使用SQLiteDatabase.execSQL(String sql, Object[] bindArgs)方法来实现,具体如下:

1     /*
2      * Function  :    添加数据
3      * Author    :    博客园-pres_cheng
4      */
5     public void addStudentInfo(Student student) {
6         db = mySQLiteOpenHelper.getWritableDatabase();
7         db.execSQL("INSERT INTO tab_student (studentId, studentName, studentAge) values (?, ?, ?)", 
8           new Object[] {student.getStudentId(), student.getStudentName(), student.getStudentAge()});
9     }

其中,通过第二个参数bindArgs,使SQL语句中的问号(?)与这个数组中的值形成一一对应关系,从而将值写入到“tab_student”表中的对应字段中。

 

5.2更新数据

  更新数据的方法与添加数据的方法大致相同,具体如下:

 

1    /*
2     * Function  :    更新数据
3     * Author    :    博客园-pres_cheng
4     */
5    public void updateStudentInfo(Student student) {
6        db = mySQLiteOpenHelper.getWritableDatabase();
7        db.execSQL("UPDATE tab_student SET studentName = ?, studentAge = ? WHERE studentId = ?", 
          new Object[] {student.getStudentName(), student.getStudentAge(), student.getStudentId()});
8    }

5.3查询数据

   查询数据时,因为需要返回查询的结果,所以需要使用SQLiteDatabase.rawQuery()方法将查询的结果返回,具体如下:

1     /*
 2      * Function  :    查询数据
 3      * Author    :    博客园-pres_cheng
 4      */
 5     public Student findStudentInfo(int id) {
 6         db = mySQLiteOpenHelper.getWritableDatabase();
 7         String sql = "SELECT studentId, studentName, studentAge FROM tab_student WHERE studentId = ?";
 8         Cursor cursor = db.rawQuery(sql, new String[] {String.valueOf(id)});
 9         if(cursor.moveToNext()) {
10             return new Student(cursor.getInt(cursor.getColumnIndex("studentId")),     
                                  cursor.getString(cursor.getColumnIndex("studentName")),
11                                cursor.getInt(cursor.getColumnIndex("studentAge")));
12         }
13         return null;
14     }

可以看出,通过使用SQLiteDatabase.rawQuery()方法可以将查询到的结果存入Cursor对象中。然后,我们可以使用Cursor对象的getXXX()方法将查询结果从Cursor对象中取出来。

  当然了,我们还可以根据实际的需要,去实现更多的接口方法,比如,删除数据、获取数据列表、获取数据个数等等。

  封装好了以上的这些接口方法,便可以很方便的在程序中直接调用这些方法,不必再去关心底层数据库的调用,而将精力放在UI界面的设计实现上。