先說說多線程 數據庫sqlite問題:
1:多線程 單例 SQLiteOpenHelper 可以同時讀和寫。
2.多線程 多個對象 SQLiteOpenHelper 不能同時讀寫(可以同時讀)。
3.多線程事務(1個讀寫過程就是一個事務,推薦以後都用事務的方式,對於大量的數據)
4.事務寫法:
1. 使用db.execSQL(sql)
- public void inertOrUpdateDateBatch(List<String> sqls) {
- SQLiteDatabase db = getWritableDatabase();
- db.beginTransaction();
- try {
- for (String sql : sqls) {
- db.execSQL(sql);
- }
- // 設置事務標誌爲成功,當結束事務時就會提交事務
- db.setTransactionSuccessful();
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 結束事務
- db.endTransaction();
- db.close();
- }
- }
2. 使用db.insert("table_name", null, contentValues)
- db.beginTransaction(); // 手動設置開始事務
- for (ContentValues v : list) {
- db.insert("bus_line_station", null, v);
- }
- db.setTransactionSuccessful(); // 設置事務處理成功,不設置會自動回滾不提交
- db.endTransaction(); // 處理完成
- db.close();
3. 使用InsertHelper類
這個類在API 17中已經被廢棄了,參考博客:http://www.outofwhatbox.com/blog/2010/12/android-using-databaseutils-inserthelper-for-faster-insertions-into-sqlite-database/
- InsertHelper ih = new InsertHelper(db, "bus_line_station");
- db.beginTransaction();
- final int directColumnIndex = ih.getColumnIndex("direct");
- final int lineNameColumnIndex = ih.getColumnIndex("line_name");
- final int snoColumnIndex = ih.getColumnIndex("sno");
- final int stationNameColumnIndex = ih.getColumnIndex("station_name");
- try {
- for (Station s : busLines) {
- ih.prepareForInsert();
- ih.bind(directColumnIndex, s.direct);
- ih.bind(lineNameColumnIndex, s.lineName);
- ih.bind(snoColumnIndex, s.sno);
- ih.bind(stationNameColumnIndex, s.stationName);
- ih.execute();
- }
- db.setTransactionSuccessful();
- } finally {
- ih.close();
- db.endTransaction();
- db.close();
- }
4. 使用SQLiteStatement
查看InsertHelper時,官方文檔提示改類已經廢棄,請使用SQLiteStatement,鏈接:https://developer.android.com/reference/android/database/DatabaseUtils.InsertHelper.html 該方法類似於JDBC裏面的預編譯sql語句,使用方法如下:
- String sql = "insert into bus_line_station(direct,line_name,sno,station_name) values(?,?,?,?)";
- SQLiteStatement stat = db.compileStatement(sql);
- db.beginTransaction();
- for (Station line : busLines) {
- stat.bindLong(1, line.direct);
- stat.bindString(2, line.lineName);
- stat.bindLong(3, line.sno);
- stat.bindString(4, line.stationName);
- stat.executeInsert();
- }
- db.setTransactionSuccessful();
- db.endTransaction();
- db.close();
下圖是以上4中方法在批量插入1萬條數據消耗的時間
可以發現第三種方法需要的時間最短,鑑於該類已經在API17中廢棄,所以第四種方法應該是最優的方法。