Моя база данных хранится в папке с активами и копируется во время выполнения. В настоящее время у меня есть простое действие, которое делает простой вызов базы данных:
DBAdapter adapter = new DBAdapter(HomeActivity.this);
final SQLiteDatabase db = adapter.getReadableDatabase();
final Cursor c = db.query("exercises", new String[] { "name" }, null,
null, null, null, null);
startManagingCursor(c);
if (c.moveToFirst())
Log.e(TAG, c.getString(0));
else
Log.e(TAG,"No dice");
И ниже мой DBAdapter.java (который расширяет открытый помощник):
public class DBAdapter extends SQLiteOpenHelper {
private static final int DB_VERSION = 1;
private static String DB_PATH = "";
private static final String DB_NAME = "gymrat.db";
private final Context myContext;
private static final String TAG = "GymRat.DBAdapter";
/**
* Constructor Takes and keeps a reference of the passed context in order to
* access to the application assets and resources.
*
* @param context
*/
public DBAdapter(Context context) {
super(context, DB_NAME, null, DB_VERSION);
this.myContext = context;
DB_PATH = "/data/data/"
+ context.getApplicationContext().getPackageName()
+ "/databases/";
}
/**
* Copies your database from your local assets-folder to the just created
* empty database in the system folder, from where it can be accessed and
* handled. This is done by transferring bytestream.
* */
private void copyDatabase() throws IOException {
InputStream myInput = myContext.getAssets().open(DB_NAME);
String outFileName = DB_PATH + DB_NAME;
OutputStream myOutput = new FileOutputStream(outFileName);
byte[] buffer = new byte[1024];
int length;
while ((length = myInput.read(buffer)) > 0) {
myOutput.write(buffer, 0, length);
}
myOutput.flush();
myOutput.close();
myInput.close();
}
/**
* Call on creating data base for example for creating tables at run time
*/
@Override
public void onCreate(SQLiteDatabase db) {
Log.e(TAG, "onCreate");
try {
copyDatabase();
} catch (IOException e) {
Log.e(TAG,"IOException copying Database");
}
}
@Override
public void onOpen(SQLiteDatabase db) {
Log.e(TAG, "onOpen");
try {
db.rawQuery("select * from exercises", null);
} catch ( SQLiteException e) {
Log.e(TAG,"DB copy didn't work");
try {
copyDatabase();
} catch (IOException e1) {
Log.e(TAG,"IOException recopying DB");
}
}
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
Log.e(TAG, "Upgrading");
}
}
Я взял этот адаптер из здесь и изменил его, поскольку он, похоже, не использует функциональность OpenHelper.
Моя проблема: Всякий раз, когда мое приложение запускается впервые после установки или после очистки данных в приложении, сначала вызывается onCreate, как и следовало ожидать, и вызывается метод copyDatabase. Проблема в том, что база данных не копируется - или, если это происходит, она немедленно перезаписывается. Вместо этого создается база данных по умолчанию, содержащая только таблицу «android meta_data», в результате чего запрос в моем действии вызывает исключение, поскольку таблица не существует.
Если я снова копирую базу данных в методе onOpen, он работает нормально. Я делаю что-то не в порядке или пропускаю какой-то вызов, который вызывает SQLiteOpenHelper для создания базы данных по умолчанию? Созданная база данных по умолчанию использует то же имя и номер версии, которые указаны в конструкторе.
Прямо сейчас, как вы можете видеть, я прибегаю к использованию фиктивного запроса в onOpen, чтобы увидеть, существует ли ожидаемая таблица и, если нет, продолжить копирование базы данных. Другой вариант - просто установить флаг, когда onCreate вызывается сигналом onOpen для копирования базы данных. Очевидно, они оба немного хакерские, и мне действительно любопытно, что происходит.
Я планирую перенести фактические вызовы базы данных в отдельные вспомогательные классы вне Activity, я просто вызывал базу данных напрямую, чтобы протестировать ее.