java - 如何通过查询方法(SQLiteDatabase.query)使用此查询?

标签 java android sqlite android-sqlite

我的任务是编写一个查询来返回孤立行的数量。我已经实现了这一点,但另一个任务是不使用 rawQuery 方法来使用查询方法获得相同的结果。

问题是我得到 java.lang.IllegalStateException: Invalid table

表格,有3个

  1. 具有 _id 列和 name 列的父表
  2. 子表,其中包含 _id 列、name 列和 childtoparentlink 列,该列是链接到父表的整数。
  3. 好友表,包含 _id 列、name 列和friendtochildlink 列。

创建行并将其放入表中的 SQL(包括一些孤立行)类似于

CREATE TABLE parent(_id INTEGER PRIMARY KEY,parentname TEXT);
CREATE TABLE child(_id INTEGER PRIMARY KEY,childname TEXT, childtoparentlink INTEGER);
CREATE TABLE friend(_id INTEGER PRIMARY KEY,friendname TEXT, friendtochildlink INTEGER);
INSERT INTO parent VALUES(null,'Parent A');
INSERT INTO parent VALUES(null,'Parent B');
INSERT INTO child VALUES(null,'Child A',1);
INSERT INTO child VALUES(null,'Child B',2);
INSERT INTO child VALUES(null,'Child X',10); -- orphan
INSERT INTO friend VALUES(null,'Friend A',1);
INSERT INTO friend VALUES(null,'Friend B',2);
INSERT INTO friend VALUES(null,'Friend X',100); -- orphan

使用 rawQuery 时有效并给出正确值的查询是

SELECT 
    (
        SELECT count() FROM child 
            LEFT JOIN parent ON child.childtoparentlink = parent._id 
        WHERE parent.parentname IS NULL
    ) AS child_mismatches,
    (
        SELECT count() FROM friend 
            LEFT JOIN child ON friend.friendtochildlink = child._id 
        WHERE child.childname IS NULL
    ) AS friend_mismatches
  • 我得到两列,每列的值为 1(根据需要)。

我的实际代码是:-

public ArrayList<String> checkLinkIntegrity() {
    ArrayList<String> return_value = new ArrayList<>();
    String suffix = "_mismatches";
    String child_result_cl = TB_CHILD + suffix;
    String sq_child_mismatches = "(SELECT count() FROM " +
            TB_CHILD +
            " LEFT JOIN " + TB_PARENT +
            " ON " + TB_CHILD + "." + CL_CHILDTOPARENTLINK + " = " +
            TB_PARENT + "." + CL_PARENTID +
            " WHERE " + TB_PARENT + "." + CL_PARENTNAME + " IS NULL)" +
            " AS " + child_result_cl;
    String friend_result_cl = TB_FRIEND + suffix;
    String sq_friend_mismatches = "(SELECT count() FROM " +
            TB_FRIEND +
            " LEFT JOIN " + TB_CHILD +
            " ON " + TB_FRIEND + "." + CL_FRIENDTOCHILDLINK + " = " +
            TB_CHILD + "." + CL_CHILD_ID +
            " WHERE " + TB_CHILD + "." + CL_CHILDNAME + " IS NULL)" +
            " AS " + friend_result_cl;
    String full_query = "SELECT " + sq_child_mismatches + "," + sq_friend_mismatches;
    SQLiteDatabase db = this.getWritableDatabase();
    Cursor csr;
    Log.d("RAWQUERYSQL",full_query);
    csr = db.rawQuery(full_query,null);
    return_value.addAll(dumpCursorToStringArrayList(csr,"RAWQUERY"));

    // Fails invalid table
    csr = db.query(null,new String[]{sq_child_mismatches,sq_friend_mismatches},null,null,null,null,null);
    return_value.addAll(dumpCursorToStringArrayList(csr,"SECONDTRY"));
    csr.close();
    return return_value;
}

并且 dumpCursortoStringArrayList 方法是:-

private ArrayList<String> dumpCursorToStringArrayList(Cursor csr, String tablename) {
    ArrayList<String> rv = new ArrayList<>();
    int original_position = csr.getPosition();
    csr.moveToPosition(-1);
    rv.add("Table: " + tablename);
    while (csr.moveToNext()) {
        rv.add("\tRow # " + String.valueOf(csr.getPosition() + 1));
        for (String column: csr.getColumnNames()) {
            rv.add("\t\tColumn: " + column + "\tvalue is: \t" + csr.getString(csr.getColumnIndex(column)));
        }
    }
    csr.moveToPosition(original_position);
    return rv;
}

如果我尝试“”而不是 null,我会得到相同的错误,例如

如果我只使用 rawQuery 我得到

04-22 07:07:33.914 6271-6271/s.q001 I/RESULTS: Table: RAWQUERY
04-22 07:07:33.914 6271-6271/s.q001 I/RESULTS:  Row # 1
04-22 07:07:33.914 6271-6271/s.q001 I/RESULTS:      Column: child_mismatches    value is:   1
04-22 07:07:33.914 6271-6271/s.q001 I/RESULTS:      Column: friend_mismatches   value is:   1

这是来自使用

    ArrayList<String> results = DBOpenHelper.checkLinkIntegrity();
    for (String s : results) {
        Log.i("RESULTS",s);
    }

如何使用 query 方法而不是 rawQuery 方法运行查询,以获得更好的分数?

最佳答案

您的问题是查询方法需要一个表,然后根据它生成 SQL

SELECT your_columns FROM the_table; 

由于没有表,因此会发出无效表异常。

您必须提供一些能够满足 FROM 子句的内容,它不能是什么都没有。

您可以通过几种方式解决这个问题,我想这就是家庭作业试图让您确定/探索的内容。

修复 1

您可以提供现有的表之一,例如使用

csr = db.query(null,new String[]{sq_child_mismatches,sq_friend_mismatches},null,null,null,null,null,"1");
  • 请注意第 8 个参数,此LIMIT将生成的行数限制为 1,因为表中的每一行都会生成一行。

修复2

或者由于 FROM 可以是子查询(见图),您可以使用子查询,例如你拥有的一个 enter image description here

所以你可以使用:-

    csr = db.query(
            sq_child_mismatches, //<<<<<<<<<< the fake subquery
            new String[]{
                    sq_child_mismatches,
                    sq_friend_mismatches
            },
            null    ,null,null,null,null
    );
  • 在这种情况下,由于假子查询返回单个行/结果,因此不需要 LIMIT 子句。

关于java - 如何通过查询方法(SQLiteDatabase.query)使用此查询?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55786804/

相关文章:

Android:扫描目录并显示图片(缩略图)(图片不存储在媒体存储中)

android - 在 Horizo​​ntal RecyclerView 上相对于中心设置 Item Alpha

ios - SQLite 数据库或 CoreData Store 是否可以捆绑在 iOS 应用程序中并且仍然可以修改?

java - 如何在 java servlet 中处理并发

java - 如何使用 javax.sound.sampled 包同时播放一组频率(和弦)

java - 与 弹出框架相关

android - AOSP 内核与 Stock 内核之间的技术差异

sqlite - 对整个 SQLite 表进行排序

android - 将 SQLite 数据库从 sdcard 导入到 android - 没有 Assets 文件夹

java - 在 Java 中使用什么数据类型来匹配间隔?