SQLiteOpenHelper.getWriteableDatabase() Исключение null указателя на Android
Мне удавалось использовать SQLite с прямым прямым SQL в Android, но это первый раз, когда я обертываю DB в ContentProvider. При вызове getWritableDatabase()
или getReadableDatabase()
я получаю исключение нулевого указателя. Является ли это просто глупой ошибкой, которую я сделал с инициализацией в моем коде или есть большая проблема?
public class DatabaseProvider extends ContentProvider {
...
private DatabaseHelper databaseHelper;
private SQLiteDatabase db;
...
@Override
public boolean onCreate() {
databaseHelper = new DatabaseProvider.DatabaseHelper(getContext());
return (databaseHelper == null) ? false : true;
}
...
@Override
public Uri insert(Uri uri, ContentValues values) {
db = databaseHelper.getWritableDatabase(); // NULL POINTER EXCEPTION HERE
...
}
private static class DatabaseHelper extends SQLiteOpenHelper {
public static final String DATABASE_NAME = "cogsurv.db";
public static final int DATABASE_VERSION = 1;
public static final String[] TABLES = {
"people",
"travel_logs",
"travel_fixes",
"landmarks",
"landmark_visits",
"direction_distance_estimates"
};
// people._id does not AUTOINCREMENT, because it set based on server people.id
public static final String[] CREATE_TABLE_SQL = {
"CREATE TABLE people (_id INTEGER PRIMARY KEY," +
"server_id INTEGER," +
"name VARCHAR(255)," +
"email_address VARCHAR(255))",
"CREATE TABLE travel_logs (_id INTEGER PRIMARY KEY AUTOINCREMENT," +
"server_id INTEGER," +
"person_local_id INTEGER," +
"person_server_id INTEGER," +
"start DATE," +
"stop DATE," +
"type VARCHAR(15)," +
"uploaded VARCHAR(1))",
"CREATE TABLE travel_fixes (_id INTEGER PRIMARY KEY AUTOINCREMENT," +
"datetime DATE, " +
"latitude DOUBLE, " +
"longitude DOUBLE, " +
"altitude DOUBLE," +
"speed DOUBLE," +
"accuracy DOUBLE," +
"travel_mode VARCHAR(50), " +
"person_local_id INTEGER," +
"person_server_id INTEGER," +
"travel_log_local_id INTEGER," +
"travel_log_server_id INTEGER," +
"uploaded VARCHAR(1))",
"CREATE TABLE landmarks (_id INTEGER PRIMARY KEY AUTOINCREMENT," +
"server_id INTEGER," +
"name VARCHAR(150)," +
"latitude DOUBLE," +
"longitude DOUBLE," +
"person_local_id INTEGER," +
"person_server_id INTEGER," +
"uploaded VARCHAR(1))",
"CREATE TABLE landmark_visits (_id INTEGER PRIMARY KEY AUTOINCREMENT," +
"server_id INTEGER," +
"person_local_id INTEGER," +
"person_server_id INTEGER," +
"landmark_local_id INTEGER," +
"landmark_server_id INTEGER," +
"travel_log_local_id INTEGER," +
"travel_log_server_id INTEGER," +
"datetime DATE," +
"number_of_questions_asked INTEGER," +
"uploaded VARCHAR(1))",
"CREATE TABLE direction_distance_estimates (_id INTEGER PRIMARY KEY AUTOINCREMENT," +
"server_id INTEGER," +
"person_local_id INTEGER," +
"person_server_id INTEGER," +
"travel_log_local_id INTEGER," +
"travel_log_server_id INTEGER," +
"landmark_visit_local_id INTEGER," +
"landmark_visit_server_id INTEGER," +
"start_landmark_local_id INTEGER," +
"start_landmark_server_id INTEGER," +
"target_landmark_local_id INTEGER," +
"target_landmark_server_id INTEGER," +
"datetime DATE," +
"direction_estimate DOUBLE," +
"distance_estimate DOUBLE," +
"uploaded VARCHAR(1))"
};
public DatabaseHelper(Context context) {
super(context, DATABASE_NAME, null, DATABASE_VERSION);
Log.v(Constants.TAG, "DatabaseHelper()");
}
@Override
public void onCreate(SQLiteDatabase db) {
Log.v(Constants.TAG, "DatabaseHelper.onCreate() starting");
// create the tables
int length = CREATE_TABLE_SQL.length;
for (int i = 0; i < length; i++) {
db.execSQL(CREATE_TABLE_SQL[i]);
}
Log.v(Constants.TAG, "DatabaseHelper.onCreate() finished");
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
for (String tableName : TABLES) {
db.execSQL("DROP TABLE IF EXISTS" + tableName);
}
onCreate(db);
}
}
}
Как всегда, спасибо за помощь!
-
Не уверен, что эта деталь помогает, но здесь LogCat показывает исключение:
удалена мертвая ссылка ImageShack
Ответы
Ответ 1
Вместо
databaseHelper = new DatabaseProvider.DatabaseHelper(getContext());
попробовать:
databaseHelper = new DatabaseProvider.DatabaseHelper(this);
и посмотрите, помогает ли это.
Если это не так, можете ли вы опубликовать трассировку стека NullPointerException
?
Ответ 2
Эта ошибка почти наверняка связана с недопустимым контекстом:
super(context, DATABASE_NAME, null, DATABASE_VERSION);
Ответ 3
У меня была эта проблема с SQLite nullpointerexception в неактивности, и проблема заключалась в том, что у нее был недопустимый контекст.
Мое исправление передавало контекст в onCreate вместо конструктора. Попробуйте найти контекст из другого места, чтобы сделать getApplicationContext()
.
Ответ 4
попробуйте использовать getWritableDatabase()
через некоторое время после инициализации помощника
вы получили NullPointerException
, потому что база данных не была полностью подготовлена
и не делайте этого в нити ui, или это может вызвать ANR
Ответ 5
У меня была эта проблема, и я понял, что это была красная селедка.
Моя проблема связана с тем, как я использовал ContentProvider.
Я действительно обращался к ContentProvider напрямую, а не через ContentResolver. Это, по сути, простой способ отключиться, если ваш новый для использования ContentProviders.
Надеюсь, что помогает:)
Ответ 6
Убедитесь, что при создании AVD укажите некоторое количество места на карте SD. У меня была та же проблема, и это решило это. Если это не поможет, вы также можете попробовать запустить эмулятор с параметром -wipe-data.
Ответ 7
Я получил ту же ошибку NullPointerException, когда getReadableDatabase() получает вызов внутри запроса(), потому что я неправильно запрашивал у своего поставщика контента. Я не вижу ничего плохого в том, как вы строите свою базу данных. getContext() должен предоставить вам соответствующий объект контекста. Однако я хотел бы обратить внимание на то, как вы делаете свои запросы поставщику контента. Вы должны убедиться, что у вас есть действительная ссылка на ваш контент-резольвер (т.е. Путем вызова getContentResolver(). Query (..)) и использует это для выполнения запроса.