• 使用Sqlite数据库存储数据


    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界面的设计实现上。

  • 相关阅读:
    ubuntu下按安装lamp
    linux 一些简记
    编写shell脚本步骤
    C++ array vector 数组
    抓取网页扒图片相对路径改绝对路径
    静态html文件js读取url参数
    IE7的web标准之道——1:前言(兼目录) (很牛的CSS书籍)
    感谢放逐自由《博客园精华集》分类索引
    这篇文章证实了索引对于IN,LIKE的优化程度,顺便学会了怎么看看语耗费的时间
    命名空间“System”中不存在类型或命名空间名称“Linq”(是缺少程序集引用吗?)
  • 原文地址:https://www.cnblogs.com/prescheng/p/6058178.html
Copyright © 2020-2023  润新知