1

同じクラスの異なるコンテキストでデータベース全体を再初期化する原因となる 2 つの異なるアクティビティから参照されたときに、データベースに関するいくつかのメソッドを動作させようとしています。これは 1 つのアクティビティから機能しますが、データベースが閉じられ、初期化され、他のアクティビティで開かれると、IllegalArgumentException が発生します。データベースに直接接続されているメソッドを変更していないため、これは私には意味がありません。コードが 2 つのクラスに分離されていても問題なく動作していました。

スタックトレース:

05-09 20:44:42.661: E/AndroidRuntime(953): FATAL EXCEPTION: main
05-09 20:44:42.661: E/AndroidRuntime(953): java.lang.RuntimeException: Unable to resume activity {maturaarbeit.nicola_pfister.marks/maturaarbeit.nicola_pfister.marks.Marks}: java.lang.IllegalArgumentException: the bind value at index 1 is null
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2575)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:2603)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2089)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread.access$600(ActivityThread.java:130)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1195)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.os.Handler.dispatchMessage(Handler.java:99)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.os.Looper.loop(Looper.java:137)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread.main(ActivityThread.java:4745)
05-09 20:44:42.661: E/AndroidRuntime(953):  at java.lang.reflect.Method.invokeNative(Native Method)
05-09 20:44:42.661: E/AndroidRuntime(953):  at java.lang.reflect.Method.invoke(Method.java:511)
05-09 20:44:42.661: E/AndroidRuntime(953):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:786)
05-09 20:44:42.661: E/AndroidRuntime(953):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
05-09 20:44:42.661: E/AndroidRuntime(953):  at dalvik.system.NativeStart.main(Native Method)
05-09 20:44:42.661: E/AndroidRuntime(953): Caused by: java.lang.IllegalArgumentException: the bind value at index 1 is null
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.database.sqlite.SQLiteProgram.bindString(SQLiteProgram.java:164)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.database.sqlite.SQLiteProgram.bindAllArgsAsStrings(SQLiteProgram.java:200)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.database.sqlite.SQLiteDirectCursorDriver.query(SQLiteDirectCursorDriver.java:47)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.database.sqlite.SQLiteDatabase.rawQueryWithFactory(SQLiteDatabase.java:1314)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.database.sqlite.SQLiteDatabase.queryWithFactory(SQLiteDatabase.java:1161)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase.java:1032)
05-09 20:44:42.661: E/AndroidRuntime(953):  at maturaarbeit.nicola_pfister.marks.database.DBAdapter.getAverage(DBAdapter.java:200)
05-09 20:44:42.661: E/AndroidRuntime(953):  at maturaarbeit.nicola_pfister.marks.MyMenu.getData(MyMenu.java:71)
05-09 20:44:42.661: E/AndroidRuntime(953):  at maturaarbeit.nicola_pfister.marks.Marks.onResume(Marks.java:52)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1184)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.Activity.performResume(Activity.java:5082)
05-09 20:44:42.661: E/AndroidRuntime(953):  at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2565)
05-09 20:44:42.661: E/AndroidRuntime(953):  ... 12 more

マークの例外スロー方法 (menu.onResume):

private final Context context;

private DatabaseHelper DBHelper;
private SQLiteDatabase db;

MyMenu menu = new MyMenu(this);

    ListView lView;
    View view;

    private String TAG = "Marks";
    @Override
        protected void onResume() {
            super.onResume();
            menu.closedb();
            menu.opendb(TAG);
            menu.getData(lView, view);

        }

MyMenu に含まれるメソッド (db.open() の例外):

DBAdapter db;

public void opendb(String caller) {
        db = new DBAdapter(context);
        this.caller = caller; //Is used to distinguish which activity called the class; not involved in this problem
        db.open();

    }

    public void closedb() {
        db.clean();
        db.close();
    }

@SuppressWarnings("deprecation")
public String getData(final ListView lView, final View view) {
    SimpleCursorAdapter adapter = null;
    if (caller == "Main") {
        adapter = new SimpleCursorAdapter(context, 
                android.R.layout.simple_list_item_activated_1, 
                db.getAllSubjects(), 
                new String[] { "subject" }, 
                new int[] { android.R.id.text1 });
    } else if (caller == "Marks") {
        adapter = new SimpleCursorAdapter(context, 
                android.R.layout.simple_list_item_activated_1, 
                db.getMarks(subject), 
                new String[] {"value"}, 
                new int[] { android.R.id.text1 });

        Cursor cursor = db.getAverage(selection);
        average = cursor.getDouble(cursor.getColumnIndexOrThrow(DBAdapter.KEY_VALUE));

        if (average != 0.0) {
            TextView text = new TextView(context);
            text =  (TextView)view.findViewById(R.id.marks_average);
            text.setText(context.getString(R.string.average) + " " + average);
            text.setBackgroundColor(context.getResources().getColor(R.color.lightgrey));
        } else {
            TextView text = new TextView(context);
            text =  (TextView)view.findViewById(R.id.marks_average);
            text.setText("");
            text.setBackgroundColor(context.getResources().getColor(android.R.color.transparent));
        }
    }

DBAdapter に含まれるメソッド (1 つのアクティビティで正常に動作し、getAverage() で例外をスローします):

public DBAdapter open() throws SQLException {
        db = DBHelper.getWritableDatabase();
        return this;
    }

    public void close() {
        DBHelper.close();
    }

    public boolean clean () {
        db.delete(DATABASE_TABLE_SUBJECTS, KEY_SUBJECT +"=?", new String[] {""});
        db.delete(DATABASE_TABLE_MARKS, KEY_VALUE + "=?", new String[] {""});
        return true;
    }

    public Cursor getAverage(String subject) throws SQLException {
    Cursor mCursor =
            db.query(true, DATABASE_TABLE_AVERAGE, new String[] {
                    KEY_ROWID,
                    KEY_SUBJECT,
                    KEY_VALUE
            },
            KEY_SUBJECT + "=?",
            new String[] {subject}, null, null, null, null);
    if (mCursor != null) {
        mCursor.moveToFirst();
    }
    return mCursor;
}

私は当初、エラーは open() メソッドにあると考えていましたが、スタックトレースを確認すると、この時点では呼び出されていないため、 getAverage() によって例外がスローされていることがわかりました。Marks アクティビティで DBAdapter クラスを正しく初期化するために何かが欠けていると思いますが、原因を見つけることができません。パズルを解くための情報やコードがさらに必要な場合は、遠慮なくお知らせください。大変お世話になりました。

4

2 に答える 2

3

menu.getData(lView, view)メソッドを見てもいいですか?

とにかくCursor、データベース ヘルパー クラスからオブジェクトを返さないことをお勧めします。なんで?複数の呼び出しが存在する可能性があるため、データベースへの並列呼び出しが発生する可能性があります。データベースから何かが必要になるたびに、それを開いてから閉じる必要があります。アプリケーションの一部がデータベースを閉じ、他の部分がまだ処理を終えておらず、データベースのデータから何かにアクセスしようとしたり、Cursorオブジェクトからデータを取得しようとしたりすると、確実にクラッシュします。

私の解決策は、双方向からデータをマップする Application Model <=> Database モデル オブジェクトを作成することです。

例えば:

public LinkedList<Person> getPersonList(){
     Cursor result = db.query(...);

     LinkedList<Person> people = null;

     if(result.moveToFirst()){
         people = new LinkedList<Person>();
         Person person = null;

         do{
              person = new Person(); 

              person.setBirthDate(result.getInt(0));
              /*some more attributes mapping*/

              people.add(person); 
         }while(result.hasNext());
    }

    return people;
}
于 2013-05-09T21:49:05.640 に答える