我有时会在日志中发现这个错误。
无法从包含0行64列的游标窗口中读取行0、列0。
首先,讲点背景故事。我有一个应用程序在我们公司的许多设备上运行。目前,它主要运行在大约20个三星Note8设备、2个三星Note10.1设备和一些其他设备上。到目前为止,这个问题只发生在Note8中的两个设备上。在所有其他设备上,它似乎工作得很好。
该应用程序的工作原理是用户使用该应用程序收集信息、文本、照片、签名等…所有这些都作为提交表中的一行存储/插入到sqlite数据库中。submissions表有64列以满足收集的每个字段。有一个同步方法总是在运行(它是一个在可运行线程内执行的异步任务,因此即使应用程序关闭了,它仍然在后台同步数据,除非您从android任务管理器中滑动关闭它),它将数据同步到远程服务器,并检查如果需要同步,例如插入了新提交,则每10秒将开始同步该提交。当提交完成与服务器的同步并收到成功响应时,它将从设备的数据库中删除。到目前为止,它一直工作得很好,成千上万的提交已经成功同步等。然而,时不时地,我从一个或两个特定的注8平板电脑,重现这个问题的一个错误。我尝试了很多次来重新创建错误,但是当我测试它时它总是有效的,并且我尝试了所有类型的场景来测试它。
我的代码有数千行,所以我会尽量保持它的相关性。首先,这里是runnable的相关代码:

public Runnable myRunnable = new Runnable()
 {

    @Override
    public void run()
    {
       count ++;
                if(count >= 10)
                {
                    android.util.Log.w("     SYNC     ", "---------------");
                    android.util.Log.w("     SYNC     ", "Checking if sync method is busy");
                    if(!syncBusy)
                    {
                        android.util.Log.w("     SYNC     ", "Sync method OPEN");
                        doSync();//This will start doing sync.
                        count = 0;
                    }
                    else
                    {
                    android.util.Log.w("     SYNC     ", "Sync method BUSY, will try again in 10 seconds");
                    android.util.Log.w("     SYNC     ", "---------------");
                    }
                }
                if(count == 1 && !syncBusy)
                {
                    checkSubmissionsLefttoSync();
                }
       mHandler.postDelayed(myRunnable, 1000);
    }
};

然后,dosync()方法是我进行上传的地方,也是我得到错误的地方。它有上千行长,所以我不想在这里发布代码。我可以说,除了产生上述错误的一个或两个异常之外,它对所有设备都100%工作。
另外,我将发布在sync方法中实际遍历数据库的部分:
        databaseHelper = new Handler_Database(this);
        Cursor cursor2 = databaseHelper.getAllUnsyncedSubmissions();

        if(cursor2.getCount() > 0)
        {
            if(cursor2.moveToFirst())
            {

              do
                {
                    try
                    {
                       //doing lookups here and populating sync arrays
                    }
                    catch (IllegalStateException e)
                    {
                        //Do Nothing
                    }
                    catch (NullPointerException e)
                    {
                        //Do Nothing
                    }
                }
                while(cursor2.moveToNext());

            }
            else
            {

            }
        }
        else
        {

        }
         cursor2.close();
         databaseHelper.close();

我注意到了一些其他的事情,我不确定这是否是一个问题,但是,当我运行这个应用程序时,logcat输出以下行大约6或7次:
12:48:11.115 7416 7416调试sqliteopenhelper数据库版本:60
当应用程序启动时,我有自己的警告消息,这些消息只显示一次。但是,数据库版本和其他信息被多次写入logcat。这是个问题吗?这是否意味着我的数据库有某种bug,它会创建多个实例?
所有应用程序都会使用PlayStore中的签名APK进行更新。我的数据库onupgrade方法如下:
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)
{
    // Drop older table if existed
    db.execSQL("DROP TABLE IF EXISTS " + TABLE_SUBMISSIONS);

    // Create tables again
    onCreate(db);
}

这对所有其他平板电脑和设备都适用。它们都会从PlayStore自动更新,并继续正常工作。但是,为什么logcat要多次输出这些行呢?
这可能是问题的一部分,为什么我得到的错误,正如在这篇文章的开头提到的?
任何可能的见解都将非常感谢。我已经拔头发好几个星期了,在我的代码中找不到任何错误。
更新1:
我刚看到这个警告:
08:30:58.710 16745 16745警告光标窗口已满:请求分配307333字节,可用空间249426字节,窗口大小2097152字节
我想这可能是我问题的根源。对如何有效解决这个问题有什么想法吗?
更新2:
我认为一个解决方案可能是限制使用光标从中检索的列。例如,所有小文本值/列。然后,为我知道占用了太多空间的每一列创建一个新查询。你认为这是个解决办法吗?有人吗?
更新3:
这可能行得通,我想在循环使用初始游标之后,我会在单独的游标中处理大字段。类似这样的东西(我刚刚写了一些psuedo代码):
Cursor c = db.rawQuery("SELECT Column1, Column2 .... FROM table " + ... , ...);

解决方案:
看看我的解决方案!https://stackoverflow.com/a/26797130/1518916

最佳答案

我完成了我的解决方案,现在效果很好。基本上,底线是在android上的sqlite数据库中存储大数据是不理想的。使用图像时,只需存储uri并将图像保存在设备上。但是,如果必须这样做,我建议不要将所有图像数据加载到同一个光标中,将它们拆分并执行多个查询。
请看我的答案https://stackoverflow.com/a/26797130/1518916

关于android - 无法从具有0行,64列的CursorWindow读取第0行第0列,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/26755590/

10-10 10:18