Хорошая схема для удаления файла базы данных в SQLiteOpenHelper.onDowngrade()
у меня есть существующая база данных, основанная на SQLiteOpenHelper
который имеет несколько версий и код для его обновления, и это отлично работает. Но в случае, если пользователь устанавливает более старую версию приложения (которая ожидает более низкую версию базы данных), он в настоящее время аварийно завершит работу -ContentProvider
использование его не может получить доступ к базе данных. Я хотел бы предотвратить его сбой, но я не хочу на самом деле понизить базу данных - добавление кода для этого будет болью. Удаление всех таблиц, безусловно, будет работать, но начиная с a свежий файл ИМО чище и менее подвержен ошибкам.
это о том, как выглядит помощник базы данных - ничего особенного
public class MyDbHelper extends SQLiteOpenHelper {
private static final int DATABASE_VERSION = 3;
private static final String DATABASE_NAME = "my.db";
public MyDbHelper(Context context) {
super(context, DATABASE_NAME, null, DATABASE_VERSION);
}
@Override
public void onCreate(SQLiteDatabase db) {
onUpgrade(db, 0, DATABASE_VERSION);
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
if (newVersion < 1) db.execSQL("CREATE TABLE A...");
if (newVersion < 2) db.execSQL("CREATE TABLE B...");
if (newVersion < 3) db.execSQL("CREATE TABLE C...");
}
@Override
public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) {
// I'd like to delete the database here
// the only problem is that I can't from here
// since this is called in the middle of getWritableDatabase()
// and SQLiteDatabase has no .recreate() method.
}
}
возможные способы, которыми я пришел, чтобы сделать это:
- сделайте это извне: поймать исключения в
ContentProvider
, удалите файл и запросите снова открыть базу данных. - Мне это не нравится, так как это не входит в обязанности провайдера. - замена
SQLiteOpenHelper
С моей собственной копией этого класса, которая удаляет файл вместо вызоваonDowngrade
- проблема в том, что он использует частные части пакетаSQLiteDatabase
(например,.lock()
) который я не могу заменить без дублированияSQLiteDatabase
тоже (это, вероятно, приведет к дублированию всего стека sqlite).
есть ли хороший подход, чтобы сделать это, или мне придется идти DROP TABLES
путь например, как описано здесь?
1 ответов
я выяснил способ, который отлично работает, расширяя SQLiteOpenHelper и все, что мне нужно сделать в MyDbHelper
расширить этот класс.
public abstract class DeletingSQLiteOpenHelper extends SQLiteOpenHelper {
private static final String TAG = DeletingSQLiteOpenHelper.class.getSimpleName();
private final File mDatabaseFile;
public DeletingSQLiteOpenHelper(Context context, String name, CursorFactory factory, int version,
DatabaseErrorHandler errorHandler) {
super(context, name, factory, version, errorHandler);
mDatabaseFile = context.getDatabasePath(name);
}
public DeletingSQLiteOpenHelper(Context context, String name, CursorFactory factory, int version) {
super(context, name, factory, version);
mDatabaseFile = context.getDatabasePath(name);
}
@Override
public synchronized SQLiteDatabase getWritableDatabase() {
try {
return super.getWritableDatabase();
} catch (SQLiteDowngradeFailedException e) {
// that's our notification
}
// try to delete the file
mDatabaseFile.delete()
// now return a freshly created database
return super.getWritableDatabase();
}
@Override
public final void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) {
// throwing a custom Exception to catch it in getWritableDatabase
throw new SQLiteDowngradeFailedException();
}
// that's the exception
static class SQLiteDowngradeFailedException extends SQLiteException {
public SQLiteDowngradeFailedException() {}
public SQLiteDowngradeFailedException(String error) {
super(error);
}
}
}