Как динамически создать новую таблицу в комнате? - PullRequest
0 голосов
/ 25 октября 2019

Помогите мне, пожалуйста.
В моем приложении пользователь может создавать новые таблицы базы данных нажатием кнопки. Пользователь также может получить доступ к этим таблицам позже, чтобы получать от них данные.
В настоящее время я использую старый добрый SQLite через класс dataBaseHelper, поэтому я мог просто

public static void createTable(String tableName) {
        databaseHelper.getWritableDatabase().execSQL("CREATE TABLE IF NOT EXISTS [" + tableName + "] (...columns)");
    }

, и работа была выполнена;

Теперь я хочу использовать библиотеку Room в своем приложении вместо DataBaseHelper.
Я смотрю на этот интерфейс DAO, который строго привязан к конкретной предопределенной таблице, и не знаю, что делать.
Есть ли способ

1 Ответ

0 голосов
/ 26 октября 2019

Есть ли способ?

Существует НО динамически добавленные таблицы, вероятно, придется использовать с помощью традиционных (предварительных) методов через SupportSQLiteDatabaseэкземпляр (это эквивалент комнаты SQLiteDatabase).

Таким образом, вы эффективно побеждаете некоторые из основных причин использования Room , такие как объектно-ориентированный подход и сокращение кода Boiler-Plate .

Пример

В следующем простом примере создается и заполняется (если новая) таблица сгенерированной / управляемой комнаты, а затем динамически создается и заполняется (если новая) другая таблица НО вне ООсторона комнаты через экземпляр SupportSQLiteDatabase. Наконец, все данные извлекаются из таблиц в Курсор, а данные сбрасываются (чтобы подтвердить концепцию).

Приложение запускается дважды, чтобы показать, что существование таблицы без номера не приводит ккомната, в которой обнаружена измененная схема, и результирующее исключение.

  • Обратите внимание, что выше не учитывается управление переменным количеством динамических таблиц, таких как хранение / получение имен таблиц динамически добавляемых таблицэто еще более усложнит ситуацию.

Код: -

BaseEntity.java

@Entity(tableName = "base")
public class BaseEntity {

    public static final String BASETABLE_NAME = "base";
    public static final String BASETABLE_COL_ID = BaseColumns._ID;
    public static final String BASETABLE_COL_VALUE = "value";
    public static final String BASETABLE_NAME_PLACEHOLDER = ":tablename:";
    public static final String BASETABLE_CREATE_SQL = "CREATE TABLE IF NOT EXISTS "
            + BASETABLE_NAME_PLACEHOLDER +
            "(" +
            BASETABLE_COL_ID + " INTEGER PRIMARY KEY," +
            BASETABLE_COL_VALUE + " TEXT)";
    @PrimaryKey
    @ColumnInfo(name = BASETABLE_COL_ID)
    Long id;
    @ColumnInfo(name = BASETABLE_COL_VALUE)
    String value;

    public BaseEntity() {}

    @Ignore
    public BaseEntity(String value) {
        this.value = value;
    }

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getValue() {
        return value;
    }

    public void setValue(String value) {
        this.value = value;
    }

    @Ignore
    public static Long insertRow(SupportSQLiteDatabase sdb, String tableName, String value) {
        ContentValues cv = new ContentValues();
        cv.put(BASETABLE_COL_VALUE,value);
        return sdb.insert(tableName, OnConflictStrategy.IGNORE,cv);
    }

    @Ignore
    public static int getTableRowCount(SupportSQLiteDatabase sdb,String tableName) {
        int rv = 0;
        Cursor csr = sdb.query("SELECT count() FROM " + tableName,null);
        if (csr.moveToFirst()) {
            rv = csr.getInt(0);
        }
        csr.close();
        return rv;
    }
}
  • Как видно, это смеськода комнаты и кода не комнаты

BaseEntityDao.java

@Dao
interface BaseEntityDao {

    @Insert
    long insertRow(BaseEntity baseEntity);

    @Query("INSERT INTO base (value) VALUES(:the_value)")
    void insertRow(String the_value);

    @Query("SELECT count() FROM base")
    Integer getRowCount();

}
  • Обработчик аннотаций комнаты требует, чтобы идентификаторы SQLite (имена таблиц, имена столбцов) были такими же, какони не могут быть переменными, и, следовательно, их можно использовать только для доступа к таблице, определенной в Комнате (отсюда необходимость в эквивалентах (в этом примере статически определенная вКласс BaseEntity)).

Database.java

@androidx.room.Database(version = 1,entities = {BaseEntity.class})
public abstract class Database extends RoomDatabase {

    public abstract BaseEntityDao baseEntityDao();
}

MainActivity.java

public class MainActivity extends AppCompatActivity {

    Database mDB;
    BaseEntityDao mDao;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mDB = Room.databaseBuilder(this,Database.class,"basedb")
                .allowMainThreadQueries()
                .build();
        mDao = mDB.baseEntityDao();
        addSomeDataViaRoom();
        String dynamicTableName = "testing";
        addTable(dynamicTableName);
        addSomeDataOutsideOfRoom(dynamicTableName);
        SupportSQLiteDatabase sdb = mDB.getOpenHelper().getWritableDatabase();
        Cursor csr = sdb.query("SELECT * FROM " + BaseEntity.BASETABLE_NAME);
        DatabaseUtils.dumpCursor(csr);
        csr = sdb.query("SELECT * FROM " + dynamicTableName);
        DatabaseUtils.dumpCursor(csr);
        mDB.close();
    }

    private boolean addTable(String tableName) {

        SupportSQLiteDatabase sdb = mDB.getOpenHelper().getWritableDatabase();
        try {
            sdb.execSQL(BaseEntity.BASETABLE_CREATE_SQL.replace(BaseEntity.BASETABLE_NAME_PLACEHOLDER, tableName));
        } catch (SQLiteException e) {
            return false;
        }
        return true;
    }

    private void addSomeDataViaRoom() {
        if (mDao.getRowCount() > 0) return;
        mDao.insertRow("A");
        mDao.insertRow("B");
        mDao.insertRow("C");
    }

    private void addSomeDataOutsideOfRoom(String tableName) {
        SupportSQLiteDatabase sdb = mDB.getOpenHelper().getWritableDatabase();
        if (BaseEntity.getTableRowCount(sdb,tableName) > 0) return;
        BaseEntity.insertRow(sdb,tableName,"X");
        BaseEntity.insertRow(sdb,tableName,"Y");
        BaseEntity.insertRow(sdb,tableName,"Z");
    }
}

Результат (2-й запуск)

2019-10-26 08:04:28.650 I/System.out: >>>>> Dumping cursor android.database.sqlite.SQLiteCursor@5322d6
2019-10-26 08:04:28.651 I/System.out: 0 {
2019-10-26 08:04:28.651 I/System.out:    _id=1
2019-10-26 08:04:28.651 I/System.out:    value=A
2019-10-26 08:04:28.651 I/System.out: }
2019-10-26 08:04:28.651 I/System.out: 1 {
2019-10-26 08:04:28.651 I/System.out:    _id=2
2019-10-26 08:04:28.651 I/System.out:    value=B
2019-10-26 08:04:28.651 I/System.out: }
2019-10-26 08:04:28.651 I/System.out: 2 {
2019-10-26 08:04:28.651 I/System.out:    _id=3
2019-10-26 08:04:28.651 I/System.out:    value=C
2019-10-26 08:04:28.651 I/System.out: }
2019-10-26 08:04:28.651 I/System.out: <<<<<
2019-10-26 08:04:28.651 I/System.out: >>>>> Dumping cursor android.database.sqlite.SQLiteCursor@f873957
2019-10-26 08:04:28.652 I/System.out: 0 {
2019-10-26 08:04:28.652 I/System.out:    _id=1
2019-10-26 08:04:28.652 I/System.out:    value=X
2019-10-26 08:04:28.652 I/System.out: }
2019-10-26 08:04:28.652 I/System.out: 1 {
2019-10-26 08:04:28.652 I/System.out:    _id=2
2019-10-26 08:04:28.652 I/System.out:    value=Y
2019-10-26 08:04:28.652 I/System.out: }
2019-10-26 08:04:28.652 I/System.out: 2 {
2019-10-26 08:04:28.652 I/System.out:    _id=3
2019-10-26 08:04:28.652 I/System.out:    value=Z
2019-10-26 08:04:28.652 I/System.out: }
2019-10-26 08:04:28.652 I/System.out: <<<<<
...