我想创建一个包含多种类型的JSON对象。这是结构:
{
"key1": "value",
"key2": ["val", "val", "val"]
"key3": { "keyX": 12 }
}
如何制作一个接受所有这些类型的HashMap
?
我正在尝试这个:
let item = HashMap::new();
item.insert("key1", someString); //type is &str
item.insert("key2", someVecOfStrings); //type is Vec<String>
item.insert("key3", someOtherHashMap); //Type is HashMap<&str, u32>
let response = json::encode(&item).unwrap();
我知道哈希映射没有足够的类型信息,但我不确定如何使其工作。我尝试在item
上设置显式类型HashMap<&str, Encodable>
,但这只是另一个错误。这样做的正确方法是什么?
答案 0 :(得分:12)
您应该在HashMap
中使用枚举类型作为值。该枚举需要为每种可能的类型(布尔值,数字,字符串,列表,映射...)和每个变体的相应类型的相关值提供变量:
enum JsonValue<'a> {
String(&'a str),
VecOfString(Vec<String>),
AnotherHashMap(HashMap<&'a str, u32>),
}
幸运的是,已经有an implementation of a JSON value type serde_json crate的一部分,它是在serde包装箱上构建的。
如果你使用了serde_json包,你的代码就是这样的:
extern crate serde_json;
use serde_json::{Value, Map, Number};
fn main() {
let mut inner_map = Map::new();
inner_map.insert("x".to_string(), Value::Number(Number::from(10u64)));
inner_map.insert("y".to_string(), Value::Number(Number::from(20u64)));
let mut map = Map::new();
map.insert("key1".to_string(), Value::String("test".to_string()));
map.insert(
"key2".to_string(),
Value::Array(vec![
Value::String("a".to_string()),
Value::String("b".to_string()),
]),
);
map.insert("key3".to_string(), Value::Object(inner_map));
println!("{}", serde_json::to_string(&map).unwrap());
// => {"key1":"test","key2":["a","b"],"key3":{"x":10,"y":20}}
}
答案 1 :(得分:3)
这是另一种可能更适合您的方法。 serde_json
crate提供了一种从JSON文字构造serde_json::Value
个对象的方法。你的例子看起来像这样:
#[macro_use]
extern crate serde_json;
fn main() {
let item = json!({
"key1": "value",
"key2": ["val", "val", "val"],
"key3": { "keyX": 12 }
});
let response = serde_json::to_string(&item).unwrap();
}