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

Sla*_* Th 3 traits rust

我想将准备好的语句与rusqlite一起使用。ToSqlRusqlite 实现了String,&str以及许多其他类型的特征:

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();
}
Run Code Online (Sandbox Code Playgroud)

这不会编译并出现错误:

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();
}
Run Code Online (Sandbox Code Playgroud)

She*_*ter 6

编译器消息不会骗你。你有一个&[&String]不是一个&[&ToSql]特征对象是一种不同的类型,并且通常与基础类型具有不同的大小;将值打包到向量中时,两者都是重要的考虑因素。

另一个问题是您无法创建一个String、引用它,然后将其存储在变量中。将String立即释放,留下悬空引用,因此编译器会阻止这种情况发生。

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

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();
Run Code Online (Sandbox Code Playgroud)

完整示例

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

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()
}
Run Code Online (Sandbox Code Playgroud)
let _rows = cached_statement.query(&do_the_thing(&vec_values)).unwrap();
Run Code Online (Sandbox Code Playgroud)

完整示例

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

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();
Run Code Online (Sandbox Code Playgroud)

完整示例