I want to use a prepared statement with rusqlite. Rusqlite implements the trait ToSql
for String
, &str
and a bunch of other types:
The compiler message isn't lying to you. You have a &[&String]
not a &[&ToSql]
. A trait object is a different type and often a different size from the underlying type; both are important considerations when packing values into a vector.
Another problem is that you cannot create a String
, take a reference to it, then store that in a variable. The String
would be deallocated immediately, leaving a dangling reference, so the compiler prevents that.
The easiest thing you can do is to create a new Vec
that contains the trait object references:
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();
(complete example)
Or if you wanted an overly-generic function to perform the conversion:
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()
}
let _rows = cached_statement.query(&do_the_thing(&vec_values)).unwrap();
(complete example)
In many cases, you can use the params! or named_params! macro:
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();
(complete example)