首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >无法调用rusqlite的查询,因为它需要类型&[&rusqlite::type::ToSql]

无法调用rusqlite的查询,因为它需要类型&[&rusqlite::type::ToSql]
EN

Stack Overflow用户
提问于 2017-10-07 20:03:22
回答 1查看 1.3K关注 0票数 3

我想在粗粒岩中使用一个预先准备好的语句。Rusqlite实现了用于ToSqlString&str一堆其他类型的的特性

代码语言:javascript
复制
extern crate rusqlite;

use rusqlite::Connection;

fn main() {
    let mut connection = Connection::open("C:\\test_db.db").unwrap();

    let mut cached_statement = connection
        .prepare_cached("SELECT ?, ?, ? FROM test")
        .unwrap();

    let vec_values = vec![
        &"test1".to_string(),
        &"test2".to_string(),
        &"test3".to_string(),
    ];

    let rows = cached_statement.query(vec_values.as_slice()).unwrap();
}

这不使用错误进行编译:

代码语言:javascript
复制
error[E0308]: mismatched types
  --> src/main.rs:18:39
   |
18 |     let rows = cached_statement.query(vec_values.as_slice()).unwrap();
   |                                       ^^^^^^^^^^^^^^^^^^^^^ expected trait rusqlite::types::ToSql, found struct `std::string::String`
   |
   = note: expected type `&[&rusqlite::types::ToSql]`
              found type `&[&std::string::String]`
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2017-10-07 23:01:09

编译器的消息并没有对你撒谎。你有一个&[&String],而不是&[&ToSql]。属性对象是与基础类型不同的类型,通常是不同的大小;在将值打包到向量时,两者都是重要的考虑事项。

另一个问题是,您不能创建一个String,对它进行引用,然后将其存储在一个变量中。String将立即被释放,留下一个悬空的引用,因此编译器将防止这种情况发生。

您可以做的最简单的事情是创建一个包含特征对象引用的新Vec

代码语言:javascript
复制
let vec_values = vec![
    "test1".to_string(),
    "test2".to_string(),
    "test3".to_string(),
];

let query_values: Vec<_> = vec_values.iter().map(|x| x as &dyn ToSql).collect();

let _rows = cached_statement.query(&query_values).unwrap();

(完整示例)

或者,如果您想要一个过于通用的函数来执行转换:

代码语言:javascript
复制
fn do_the_thing<'a, I, T: 'a>(things: I) -> Vec<&'a dyn ToSql>
where
    I: IntoIterator<Item = &'a T>,
    T: ToSql,
{
    things.into_iter().map(|x| x as &dyn ToSql).collect()
}
代码语言:javascript
复制
let _rows = cached_statement.query(&do_the_thing(&vec_values)).unwrap();

(完整示例)

在许多情况下,您可以使用params!named_params!宏:

代码语言:javascript
复制
let a = "test1".to_string();
let b = "test2".to_string();
let c = "test3".to_string();

let _rows = cached_statement.query(params![a, b, c]).unwrap();

(完整示例)

票数 6
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/46624591

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档