1.SQLite数据库的好处

     SQLite是轻量级的数据库,是嵌入式的,不需要安装,启动服务器进程,而且也支持事物管理,占用的系统资源也比较少,专门适用于在资源有限的设备上适量数据存取。


     用官方的话来说:当所有的功能被打开的时候,根据目标平台和编译器的优化设置,库文件的大小可以小于500k(64位码更大,有的编译器优化也会导致更大),如果省略可选功能,可以减少到低于300k,SQLite占用的堆栈空间只有4~100k,在内存使用和速度上面有一个平衡。总之就是一个劲的跨它小而不失性能,简而不缺功能。




2.创建SQLite数据库


  看sqlitedabase的api上面的方法创建:

openOrCreateDatabase(File file, CursorFactory factory);   //factory传null是调用默认的CursorFactory 
  openOrCreateDatabase(String path, CursorFactory factory);
  openOrCreateDatabase(String path, CursorFactory factory, DatabaseErrorHandler errorHandler);


 但是为了数据版本的升级和维护一般我们用的都是SQLiteOpenHelper创建,所以上面大家了解就够了。


3.数据库的操作 


execSQL(String sql, Object[] bindArgs);    //增删改,返回指为null,开始我还一直想为啥查询不也用这个,逗比的发现没返回值,查出来也得不到...
  rawQuery(String sql, String[] selectionArgs) ; //用于查询操作,返回的cursor(下面有简单介绍cursor)

  //android提供很多方法来操作数据库,但是还是建议利用上面的方法,毕竟写数据库语句都是通用。
  insert(String table, String nullColumnHack, ContentValues values);
  query(String table, String[] columns, String selection,
            String[] selectionArgs, String groupBy, String having,
            String orderBy, String limit) ;          //api里面提供很多,不一一列出。
  replace(String table, String nullColumnHack, ContentValues initialValues);
  delete(String table, String whereClause, String[] whereArgs);
  update(String table, ContentValues values, String whereClause, String[] whereArgs) ;




4.什么是cursor


  源码里面给出的说明是:这个接口提供随机读写访问一个数据库查询返回的结果集。


  网上给的说明:Cursor 是每行的集合。


                           使用 moveToFirst() 定位第一行。 


你必须知道每一列的数据类型。 


                            Cursor 是一个随机的数据源。


                            所有的数据都是通过下标取得。


  我给出代码吧:


Cursor c = db.rawQuery( "SELECT * from tablename where group_id = ? " , new String[] { group_id });
  while(c.moveToNext())
  {
        //假设第一列的列名为name,下面是两种取name数据的方式
       String name = c.getString(c.getColumnIndex("name"));    //这就是为什么要知道每列的数据类型和名称了
       String name2 = c.getString(0);    //不知道名称就要知道他是第几列(还是建议写上面的吧,方便维护代码)
   }
   c.close();  //一定要关闭cursor,不然会很悲剧的...



5.事物处理



   beginTransaction();               //手动设置事务开始,启动模式EXCLUSIVE mode



    beginTransactionNonExclusive();   //启动模式 IMMEDIATE mode



    setTransactionSuccessful();    //设置事务处理成功,不设置会自动回滚不提交。



    inTransaction();                   //判断是否在事务处理中,返回类型为boolean



    endTransaction();                  //处理完成



6.SQLiteOpenHelper的升级或降低数据库版本原理。
   我看的项目涉及到数据库,有直接在项目里面导入数据库的,有用内容提供者操作的,但是我在上家公司写的数据库还是用的SQLiteOpenHelper。
    SQLiteOpenHelper的构造方法:
   SQLiteOpenHelper(Context context, String name, CursorFactory factory, int version);
   从源码上面来在调用getWritableDatabase和getReadableDatabase会执行getDatabaseLocked方法,在此方法中
有此代码,当版本高于数据库原版本则调用onUpgrade(),当小于原版本则调用onDowngrade():



final int version = db.getVersion();
            if (version != mNewVersion) {
                if (db.isReadOnly()) {
                    throw new SQLiteException("Can't upgrade read-only database from version " +
                            db.getVersion() + " to " + mNewVersion + ": " + mName);
                }
                db.beginTransaction();
                try {
                    if (version == 0) {
                        onCreate(db);
                    } else {
                        if (version > mNewVersion) {
                            onDowngrade(db, version, mNewVersion);
                        } else {
                            onUpgrade(db, version, mNewVersion);
                        }
                    }
                    db.setVersion(mNewVersion);
                    db.setTransactionSuccessful();
                } finally {
                    db.endTransaction();
                }
            }


7.SQLiteOpenHelper类的具体实现代码。


public class DBHelper extends SQLiteOpenHelper {

 public static final String TABLE_NAME= "mytable";

 private static final int CURRENT_VERSION = 1;  //每次升级的时候数字加1

 public DBHelper(Context context, String name, CursorFactory factory) {
  //name为数据库名,null为默认的CursorFactory
  super(context, name , null, CURRENT_VERSION);
 }

 @Override
 public void onCreate(SQLiteDatabase db) {
  db.execSQL("CREATE table IF NOT EXISTS "
    + TABLE_NAME
    + " (_id INTEGER PRIMARY KEY AUTOINCREMENT, name TEXT, img TEXT,time TEXT,num TEXT,"
    + "message TEXT, group_id TEXT, isgroup TEXT, other_id TEXT)");
 }

 @Override
 public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
 //看上面第6部分贴的源码可知oldVersion即db.getVersion();(得到是已存在的数据库版本),newVersion为升级后的版本号。
  switch (newVersion) {
  case 2:
            if(oldVersion<2)
            {
                //执行从版本1升级到版本2操作
             }
  case 3:
            if(oldVersion<3)
            {
                //执行从版本2升级到版本3操作
             }
  case 4:
           break;
  }
 }
}





简单的写下数据库操作类:



    

public class GroupInfoDB {
 private DBHelper helper;

 public GroupInfoDB(Context context,String name,) {
  helper = new DBHelper(context,name ,null);
 }
 
 public void addGroup(GroupInfoBean entity) {
  SQLiteDatabase db = helper.getWritableDatabase(); 
  //建议用sql语句写,本例只是个demo
  ContentValues values = new ContentValues();
  values.put("group_id", entity.getGroup_id());
  db.insert(DBHelper.TABLE_NAME, null, values);
 }
 public void close() {
  if (db.isOpen())
   db.close();
 }
}










8.getWritableDatabase与getReadableDatabase的区别



     getWritableDatabase() 方法以读写方式打开数据库,一旦数据库的磁盘空间满了,数据库就只能读而不能写,倘若使用的是getWritableDatabase() 方法就会出错。



         getReadableDatabase()方法则是先以读写方式打开数据库,如果数据库的磁盘空间满了,就会打开失败,当打开失败后会继续尝试以只读方式打开数据库。如果该问题成功解决,则只读数据库对象就会关闭,然后返回一个可读写的数据库对象。