SQLiteOpenHelper onCreate не создает базу данных - или она немедленно перезаписывается - PullRequest
4 голосов
/ 20 марта 2011

Моя база данных хранится в папке с активами и копируется во время выполнения. В настоящее время у меня есть простое действие, которое делает простой вызов базы данных:

    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, я просто вызывал базу данных напрямую, чтобы протестировать ее.

1 Ответ

5 голосов
/ 21 апреля 2011

Если вы хотите скопировать базу данных из ресурсов при первом запуске и открыть базу данных из data / data / program.name / database, вы должны использовать код примерно так:

public class YourDatabase extends SQLiteOpenHelper

public YourDatabase(Context context) {
    super(context, DatabaseName, null, DATABASE_VERSION);
    this.dbContext = context;
    // checking database and open it if exists        
    if (checkDataBase()) {
    openDataBase();
    } else {
    try {
            this.getReadableDatabase();
            copyDataBase();
        this.close();
        openDataBase();

        } catch (IOException e) {
            throw new Error("Error copying database");
        }
     Toast.makeText(context, "Initial database is created", Toast.LENGTH_LONG).show();
    }
}

Полный ответ вы можете прочитать из этого вопроса

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...