1

我使用 soci 库来访问 sqlite3 数据库:

rowset<row> rs = (sql.prepare << "select * from my_table");
for (auto it = rs.begin(); it != rs.end(); ++it) {
    int some_value = it->get<int>(1);
    // ...
}

(这不是我的实际代码;它被大大简化以描述问题。)

现在,大多数时候,这是有效的。但是,有时,行中索引 1 处的值是字符串,而不是 int(导致std::bad_cast异常)。对应列的 SQL 类型为NUMERIC(8) NOT NULL. 我做了一些调试,看起来行中的条目数和它们的值总是正确的。唯一的问题是这一行的值有时是一个字符串。

这可能是社会中的一个错误,也可能是我这边的错误假设。我的假设是列的返回数据类型总是相同的。类型切换可能发生在我的应用程序的一次运行中 - 值可能是一个查询结果中的 int 和下一个查询结果中的字符串。

由于我不知道底层的 sqlite3 API,我什至不确定这个问题是源自 sqlite3 还是 soci。解决方法似乎很简单:检查行值的类型,如果它是字符串,则将其转换为 int。但重要的是要知道这是预期的行为还是错误。有人可以对此发表评论吗?

编辑:

像这样插入新行:

soci::session sql(Connetion::pool);
sql << "INSERT INTO " << NAME << " (REP_TIMESTAMP, REP_EVENT, REP_LTU_ID, REP_SEC_ID) VALUES (:timestamp, :event, :ltuId, :secId)",
        use (bo.timestamp),
        use (bo.event),
        use (bo.ltuId),
        use (bo.secId);

其中bo.timestampaboost::posix_time::ptime具有以下映射:

template<> struct type_conversion<boost::posix_time::ptime> {
    typedef long long base_type;
    static void from_base(const long long& t, indicator& i, boost::posix_time::ptime& target) {
        target = Helper::Time::timeFrom(t);
    }
    static void to_base(const boost::posix_time::ptime& d, long long& target, indicator& i) {
        target = Helper::Time::secondsSinceEpoch(d);
        i = i_ok;
    }
};

Helper::Time将 a 转换ptime为自 1.1.1970 以来的秒数并返回。我看不出如何REP_TIMESTAMP使用此代码将字符串放入列中。

4

1 回答 1

1

SQLite 使用动态类型;您为列声明的类型几乎没有效果。

您从数据库中获得了一个字符串值,因为您的程序将一个字符串值写入了数据库。如果要强制执行列类型,则必须添加单独的检查约束:

CREATE TABLE MyTable(
    MyColumn INTEGER NOT NULL CHECK(typeof(MyColumn) = 'integer'),
    ...
)
于 2013-07-26T08:37:15.247 回答