有什么方法可以在房间数据库中运行嵌套查询

时间:2019-12-20 10:08:15

标签: android sqlite android-room

我将旧的Android项目迁移到新标准,陷入了room database之间,无法找到如何使用SQLite执行这2条room database命令的情况。我有2个查询以嵌套方式工作,因此使用room来获得相同的结果是有可能的。

    default public ArrayList getOfflineRoomsAndObjectsByParentIds(String parentId, String site_id, Context context) {

        databaseHandler = new DatabaseHandler(context);
        Cursor cr = null;
        ArrayList<Room> offlineRoomNamesList = new ArrayList<>();
        try {

           **sqLiteDatabase = databaseHandler.getReadableDatabase();
            String query = "SELECT * FROM " + Constants.TABLE_ROOM_NAMES
                    + " WHERE " + Constants.SITE_ID + " = '" + site_id + "' AND " + Constants.PARENT_ID + "='" + parentId + "'";**

            cr = sqLiteDatabase.rawQuery(query, null);
            if (cr.moveToFirst()) {

                do {

                    Room room = new Room();
                    **room.setId(cr.getString(cr.getColumnIndex(Constants.KEY_ROOM_ID)));
                    query = "SELECT * FROM " + Constants.TABLE_OFFLINE_ROOM_OBJECTS
                            + " WHERE " + Constants.KEY_ROOM_KEY + " = ?"
                            + " AND " + Constants.SITE_ID + " = ? ORDER BY " + Constants.KEY_OBJECT_NAME;**

                    Cursor cursor = sqLiteDatabase.rawQuery(query, new String[]{cr.getString(cr.getColumnIndex(Constants.KEY_ROOM_KEY)), site_id});
                    if (cursor.moveToFirst()) {
                        do {
                            RoomObject roomObject = new RoomObject();
                            roomObject.setRoomKey(cursor.getString(cursor.getColumnIndex(Constants.KEY_ROOM_KEY)));
                            roomObject.setId(cursor.getInt(cursor.getColumnIndex(Constants.KEY_ID)));
                            roomObject.setInstanceNumber(cursor.getInt(cursor.getColumnIndex(Constants.KEY_INSTANCE_NUMBER)));
                            roomObject.setObjectType(cursor.getString(cursor.getColumnIndex(Constants.KEY_OBJECT_TYPE)));


                            String mergeKey = AppSharedPreference.getString(AppSharedPreference.MERGE_KEY, "", context);
                            if (roomObject.getObjectName().contains(mergeKey) && !TextUtils.isEmpty(mergeKey)) {
                                room.addMergeableObjectsInList(roomObject);
                            } else {
                                room.addRoomObjectsInList(roomObject);
                            }


                        }
                        while (cursor.moveToNext());
                    }
                    if (room.getRoomObjectsList().size() > 0)
                        offlineRoomNamesList.add(room);
                    cursor.close();
                } while (cr.moveToNext());
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                cr.close();
                sqLiteDatabase.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return offlineRoomNamesList;
    }

此刻我正在寻找的唯一可能性是两个定义2个不同的查询并以嵌套方式执行它们。

@Dao
public interface ControllerRoomObjectDao {



    @Query("SELECT * FROM room_names WHERE site_id=:siteId AND parentId=:parentId")
    Single<List<Room>>getOfflineRoomsAndObjectsByParentId(String siteId, String parentId);


    @Query("SELECT * FROM offline_objects WHERE site_id=:siteId AND parentId=:parentId")
    Single<List<RoomObject>>getOfflineObjectsByParentId(String siteId, String parentId);

}

TABLE room_names

TABLE offline_objects

2 个答案:

答案 0 :(得分:0)

您可以将databaseView用于您的用例 例如:

     @DatabaseView("SELECT user.id, user.name, user.departmentId," +
          "department.name AS departmentName FROM user " +
          "INNER JOIN department ON user.departmentId = department.id")

     public class UserDetail {
     public long id;
     public String name;
     public long departmentId;
     public String departmentName;
     } 

有关更多信息,https://developer.android.com/training/data-storage/room/creating-views#java

答案 1 :(得分:0)

不确定这是否完全符合您的要求,但可能会帮助具有相同一般要求的其他人。

基本上,我从this example那里得到了建议,该建议显示了如何在MySQL中执行嵌套查询/子查询。

使用相同的语法,我可以从以下方式更改我的基本DAO查询:

@Transaction
@Query("SELECT * FROM `events` WHERE `item_id` = :currentItemId AND `event_datetime` > :datetimeFrom ORDER BY `event_datetime` ASC LIMIT 1")
LiveData<Event> getNextEventWithSameItemId(long currentItemId, String datetimeFrom);

@Transaction
@Query("SELECT * FROM `events` WHERE `item_id` IN (SELECT `item_id` FROM `events` WHERE `_id` = :currentEventId) AND `event_datetime` > :datetimeFrom ORDER BY `event_datetime` ASC LIMIT 1")
LiveData<Event> getNextEventWithSameEventId(long currentEventId, String datetimeFrom);

我的事件实体在哪里

@Entity(tableName = "events")
public class Event {

    @PrimaryKey
    @ColumnInfo(name = "_id")
    public long id;

    @ColumnInfo(name = "item_id") // Relates to `items`.`_id`
    public long itemId;

    @ColumnInfo(name = "event_datetime")
    public String eventDatetime;
    
    ...
}