#android #sqlite
#Android #sqlite
Вопрос:
У меня есть следующий DBAdapter (это не все, но это соответствующий код).
private final String DATABASE_NAME = "seller";
private final String DATABASE_MY_TABLE = "sellertable";
private final Integer DATABASE_VERSION = 1
private final String CREATE_TABLE = "CREATE TABLE seller ........";
public DBAdapter( Context c )
{
this.context = c;
DBHelper = new DatabaseHelper(context);
}
private static class DatabaseHelper extends SQLiteOpenHelper
{
public DatabaseHelper( Context context )
{
super(context, DATABASE_NAME, null, DATABASE_VERSION);
}
@Override
public void onCreate( SQLiteDatabase db )
{
MyLog.d("Creating Tables");
try
{
db.execSQL(CREATE_TABLE);
}
catch (Exception e)
{
e.printStackTrace();
}
}
@Override
public void onUpgrade( SQLiteDatabase db, int oldVersion, int newVersion )
{
MyLog.w("Upgrading database from version " oldVersion " to " newVersion ". Destroying old data.");
db.execSQL("DROP TABLE IF EXISTS " DATABASE_MY_TABLE);
onCreate(db);
}
}
// Opens the database
public DBAdapter open() throws SQLException
{
db = DBHelper.getWritableDatabase();
return this;
}
Когда я пытаюсь db.open()
, я получаю следующее:
FATAL EXCEPTION: main
java.lang.RuntimeException: Unable to instantiate application com.company.app.MyApp: java.lang.NullPointerException
at android.app.LoadedApk.makeApplication(LoadedApk.java:466)
at android.app.ActivityThread.handleBindApplication(ActivityThread.java:3260)
at android.app.ActivityThread.access$2200(ActivityThread.java:117)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:969)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:130)
at android.app.ActivityThread.main(ActivityThread.java:3683)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:507)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
at dalvik.system.NativeStart.main(Native Method)
Caused by: java.lang.NullPointerException
at android.content.ContextWrapper.openOrCreateDatabase(ContextWrapper.java:203)
at android.database.sqlite.SQLiteOpenHelper.getWritableDatabase(SQLiteOpenHelper.java:118)
at com.company.app.DBAdapter.open(DBAdapter.java:74)
at com.company.app.MyApp.<init>(MyApp.java:106)
at java.lang.Class.newInstanceImpl(Native Method)
at java.lang.Class.newInstance(Class.java:1409)
at android.app.Instrumentation.newApplication(Instrumentation.java:957)
at android.app.Instrumentation.newApplication(Instrumentation.java:942)
at android.app.LoadedApk.makeApplication(LoadedApk.java:461)
Похоже, что он получает какое-то исключение, связанное с тем, когда помощник решает, создавать или обновлять базу данных или нет…
Комментарии:
1. Это просто константа для «продавца». Извините, я не ввел все свои константы, я добавлю то, что мне нужно, в приведенный выше код.
2. Это почти наверняка вызвано переданным в
Context
бытиеnull
— нет ничего другого,ContextWrapper.java:203
что могло бы вызвать NPE.3. Я считаю, что это стало проблемой, Филипп. Если вы хотите правильно ответить ниже, я отдам вам должное.
Ответ №1:
Я только что столкнулся с этим, вероятно, по другой причине. Я попытался сделать что-то связанное с БД, которое включало контекст до вызова onCreate() . Надеюсь, это кому-то поможет
Комментарии:
1. В этом причина. Принятый ответ сбит с толку. Проблема очевидна из stacktrace вопроса: база данных открывается при
MyApp.<init>
инициализации переменной-члена, задолго доonCreate()
этого.
Ответ №2:
Попробуйте добавить расширение типа «.db» в конец имени вашей базы данных.
Комментарии:
1. Ну, я исправил это другим способом.. Я не уверен, что я сделал. Я думаю, что это была проблема контекста. Я думаю, контекст приложения так и не был передан адаптеру базы данных или чему-то еще… Что происходит с .db, добавленным в конец имени? Это известное исправление или рекомендуемый способ их именования?
2. Именно так меня всегда учили. Я не знаю, имеет ли это значение, если файловой системе все равно, что расширения нет. Имя базы данных — это, после, просто имя файла.
3. как было предложено выше, лучше поместить .db
4. Это соглашение. Это не влияет ни на какой код — было до тех пор, пока вы используете то же имя.
5. обязательно посмотрите реальный ответ от @JoePlante
Ответ №3:
Если точная проблема связана с контекстом, я могу предложить ее, поскольку вчера я столкнулся с той же проблемой.
Предположим, я вызываю DatabaseHelper из другого класса, тогда мне пришлось предоставить контекст. Я был смущен тем, какой контекст: 1. getApplicationContext() 2. getBaseContext() 3. экземпляр контекста, который был инициализирован в конструкторе класса, который вызывал объект DatabaseHelper, т.е. _context.
Но фактическое решение, которое я получил, состояло в том, чтобы использовать это
Например:-
DatabaseHelper DBHelper=новый DatabaseHelper(этот);