7

我正在基于一个旧的 Android 项目构建一个 Android 应用程序。在我的新应用程序中,我使用的是 Room。我必须使用第一个项目中使用的相同数据库。此外,我使用 com.amitshekhar.android:debug-db 库从第一个项目中提取了数据库。获得数据库文件后,我想用 Room 打开它。

我正在建立这样的数据库:

Room.databaseBuilder(
            androidContext(),
            Database::class.java, "database.db"
        ).createFromAsset("database.db")
            .build()

目前我正在使用这个 createFromAsset() 方法,虽然稍后我会使用 createFromFile() 方法,因为我的数据库应该从服务器下载。

但是我得到了 java.lang.IllegalStateException: Pre-packaged database has an invalid schema 发生这种情况是因为数据库中有几种数据类型在 Room 中不受支持,例如 NVARCHAR(200)、DATE 或 bit。

我知道 Room 只使用了五种 Sql 类型,但我不知道如何更改它,以便 Room 可以使用上述方法打开这种数据库。

问题是如何将 NVARCHAR(200)、DATE 或 bit 转换为 Room 支持的数据类型?

4

3 回答 3

7

您必须将数据库转换为使用 Room 支持且与实体匹配的特定列类型关联。

对于NVARCHAR(200),您需要将NVARCHAR (200) 替换为将列定义为字符串的实体。

对于DATE它取决于实体定义,如果您使用基于字符串的日期,例如 YYYY-MM-DD hh:mm:ss 那么实体应该是 String 和列关联TEXT。如果将日期存储为时间戳,则 Entity 应该很长并且列关联 INTEGER

此处的答案 Can't migrate a table to Room do to an error with the way booleans are save in Sqlite does a conversion to change BOOL's to INTEGER。

您可以对此进行调整(尽管我对 DATE 持谨慎态度)以适应。

额外的

您可能会发现以下内容很有用。您可以在您最喜欢的 SQLite 管理器工具中针对预先存在的数据库运行它。

WITH potentialRoomChanges AS (
    SELECT sm.name AS tablename, pti.name AS columnname, pti.type, dflt_value, pk,
        CASE 
            WHEN instr(upper(pti.type),'INT') THEN 'INTEGER'
            WHEN instr(upper(pti.type),'CHAR') OR instr(upper(pti.type),'CLOB') OR instr(upper(pti.type),'TEXT') THEN 'TEXT'
            WHEN instr(upper(pti.type),'BLOB') THEN 'BLOB'
            WHEN instr(upper(pti.type),'REAL') OR instr(upper(pti.type),'FLOA') OR instr(upper(pti.type),'DOUB') THEN 'REAL'
            ELSE 'NUMERIC'
        END AS roomtype ,
        CASE WHEN pti.[notnull] THEN 'Investigate NOT NULL USE' END AS nnindicator,
        sql
    FROM sqlite_master AS sm JOIN pragma_table_info(sm.name) AS pti
    WHERE 
        sm.type = 'table' 
        AND sm.name NOT LIKE 'sqlite_%' 
        AND sm.name <> 'android_metadata' 
        AND (
            upper(pti.type) <> roomtype 
            OR instr(roomtype,'NUMERIC') 
            OR nnindicator IS NOT NULL
            OR dflt_value IS NOT NULL
            OR pk > 0
        )
    ORDER BY sm.name,pti.cid
)
SELECT tablename, columnname, type, roomtype, 
CASE WHEN upper(type) <> upper(roomtype) THEN 'Investigate TYPE should be ' ||roomtype END AS typechange_notes,
CASE WHEN roomtype = 'NUMERIC' THEN 'Investigate NUMERIC' END AS numeric_notes, 
CASE WHEN dflt_value IS NOT NULL THEN 'Investigate DEFAULT VALUE of '||dflt_value END AS default_notes,
CASE WHEN pk > 0 THEN 'Investigate PRIMARY KEY inclusion' END AS primarykey_notes,
nnindicator AS notnull_notes 
FROM potentialRoomChanges
;

示例输出:-

在此处输入图像描述

希望列/文本是不言自明的。这基于定义的列类型(可能与使用的类型不同)。例如浮点数(显示的第 5 行)你会认为是真实的。但是,根据派生类型关联,已应用第一条规则(如果类型包括 INT,则为 INTEGER)。

根据 SQLite 版本 3 - 3.1 中的数据类型的规则。柱亲和力的测定

根据我对房间的有限经验, NUMERIC不是它使用的类型,因此应始终将其更改为其他类型之一。

于 2019-07-15T15:49:44.633 回答
0

对我来说问题是Not-Null,对于数据库中的每一列,NOT NULL它都应该反映在你的模型中@NonNull

如果你LastName在数据库中是

"LastName"  TEXT NOT NULL,

在您的模型代码中,它应该是

@NonNull
private String LastName;
于 2021-05-30T14:33:17.350 回答
0

@NonNull在 Pojo(实体)类的每个字段之前使用。
无需添加@NonNull到主键字段。
下面是一个例子

@Entity(tableName = "station")
public class Station {
    @PrimaryKey
    private int id;
    @NonNull
    private String name;
    @NonNull
    private int line;
    @NonNull
    private double lat;
    @NonNull
    private double lon;

    ... constructor, getters and setters
}
于 2019-11-04T11:05:39.420 回答