Android 非法状态异常 : attempt to re-open an already-closed object: SQLiteDatabase

标签 android android-sqlite

我知道有几个像这样的问题,但它们似乎都有不同的方法来解决问题,但没有一个能解决我的问题。

我的主要 Activity 工作得很好,加载数据库并填充 ListView 。然后我调用第二个 Activity ,当我尝试加载 ListView 时,问题出现了。

我尝试过使用start/stop managingcursor(cursor)尽管它已被弃用,但它并没有解决问题。另外,我尝试在触发下一个 Activity 之前在主要 Activity 中关闭光标和数据库,但这也没有帮助。

这两个类都继承自 ListActivity 并遵循相同的顺序:

public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);        

    //Open db in writing mode
    MySQLiteHelper.init(this);
    MySQLiteHelper tpdbh =
        new MySQLiteHelper(this, "DB", null, 1);


    SQLiteDatabase db = tpdbh.getWritableDatabase();

    checkLocationAndDownloadData(); //this fires a Asynctask that calls method parseNearbyBranches shown bellow

   //I load the data to the ListView in the postExecute method of the asynctask by calling:
    /*
    Cursor cursor = MysSQLiteHelper.getBranchesNames();
    adapter = new SimpleCursorAdapter(this,
            R.layout.row, cursor, fields, new int[] { R.id.item_text },0);
    setListAdapter(adapter);
    */

    ListView lv = getListView();
    lv.setOnItemClickListener(new AdapterView.OnItemClickListener() {
     @Override
        public void onItemClick(AdapterView<?> listView, View view,
            int position, long id) {

         // Get the cursor, positioned to the corresponding row in the result set
         Cursor cursor = (Cursor) listView.getItemAtPosition(position);

       // Get the state's capital from this row in the database.
       String branch_id = cursor.getString(cursor.getColumnIndexOrThrow("branch_id"));

           cursor.close();

        openNextActivity(Integer.parseInt(branch_id));
        }
      });
}

//在另一个文件中:

private void parseNearbyBranches(JSONObject jo) throws JSONException
{
 if (   jo.has(jsonTitle) && 
            jo.has("company_id") &&
            jo.has("id")
    ) {
        String branch = jo.getString(jsonTitle);


            MySQLiteHelper tpdbh = MySQLiteHelper.instance;
            SQLiteDatabase db = tpdbh.getWritableDatabase();

            db.execSQL("INSERT INTO Branches (branch_id, name, company_id) " +
                    "VALUES ('" +jo.getInt("id")+"', '" + branch +"', '" +jo.getInt("company_id")+"' )");

            db.close(); //no difference is I comment or uncomment this line

    }
}

在 MySQLiteHelper.java 中:

public static Cursor getBranchesNames() {
        // TODO Auto-generated method stub
        String[] columns = new String[] { "_id", "branch_id", "name", "company_id" };
        Cursor c = getReadDb().query(branchesTable, columns, null, null, null, null,
                null);            
        return c;
    }

我的其他 Activity 基本相同:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_branch_surveys);

    //Read branch data from DB
        int companyID = -1;
        MySQLiteHelper.init(this);          

        String [] columns = new String [] {"company_id"};
        String [] args = {Integer.toString(branchID)};
        Cursor c = MySQLiteHelper.getReadDb().query(MySQLiteHelper.branchesTable, columns, "branch_id=?", args, null, null, null); //THIS QUERY WORKS JUST FINE

        if (c.moveToFirst())
            companyID = Integer.parseInt(c.getString(0));
        c.close();

        if( companyID != -1)
        {
            new DownloadTask().execute(Integer.toString(companyID) );
//where the Async task calls something just like NearByBranches shown above(but with different  objects of course)
//And the postExecute sets the listView:
/*  cursor = MySQLiteHelper.getAll();
    SimpleCursorAdapter adapter = new SimpleCursorAdapter(this,
            R.layout.row, cursor, fields, new int[] { R.id.item_text },0);
    setListAdapter(adapter);  
*/
        }

    }
}

在 MySQLiteHelper.java 中:

public static Cursor getAll() {
    // TODO Auto-generated method stub
    String[] columns = new String[] { "_id","title", "points" };

//********IT IS IN THIS LINE WHERE I GET THE ERROR:********************
    Cursor c = getReadDb().query(theTable, columns, null, null, null, null,
            null);

    return c;
}

  public static SQLiteDatabase getReadDb() {
        if (null == db) {
            db = instance.getReadableDatabase();
        }
        return db;
    }

希望你能帮帮我。谢谢!

最佳答案

I just tried commenting the db.close in the similar method of parseNeabyBranches and the problem was solved. Yet I dont get the same error having db.close() in parseNearbyBranches(), can you explain me why?

parseNearbyBranches() 中,您使用以下命令创建一个单独的 SQLiteDatabase 对象:

SQLiteDatabase db = tpdbh.getWritableDatabase();

由于这是一个与 getReadDb() 返回的对象不同的对象,因此您可以(并且应该)关闭它。基本规则是每次调用 getWritableDatabase()getReadableDatable() 时,都必须有一个匹配的 close() 语句。

关于Android 非法状态异常 : attempt to re-open an already-closed object: SQLiteDatabase,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15822816/

相关文章:

SQLite 将单行转换为多行

android - 点击 map 标记会导致 AlertDialog 显示所有其他 map 标记

android - 在我的应用程序中播放视频的最佳方式是什么

android - 升级到 Android 插件 3.0 后 Gradle 构建失败

java - 复制 LinearLayout View

android - 在 SQLite 错误中插入多行(错误代码 = 1)

android - 有没有办法让 Room 生成一个不自动递增的自动生成主键?

java - Android中的Sqlite : DataBaseObjectNotClosedException

java - 嵌套数据结构的对象创建,对性能的可避免影响?

java - 为什么我的 Android 应用程序将空白记录插入 MySQL 数据库?