我正在使用syn
来解析Rust代码。当我使用field.ty
读取命名字段的类型时,我得到一个syn::Type
。当我使用quote!{#ty}.to_string()
打印它时,我得到的是"Option<String>"
。
怎样才能只获取"String"
?我想在quote!
中使用#ty
来打印"String"
,而不是"Option<String>"
。
我想生成如下代码:
impl Foo {
pub set_bar(&mut self, v: String) {
self.bar = Some(v);
}
}
开始于
struct Foo {
bar: Option<String>
}
我的尝试是:
let ast: DeriveInput = parse_macro_input!(input as DeriveInput);
let data: Data = ast.data;
match data {
Data::Struct(ref data) => match data.fields {
Fields::Named(ref fields) => {
fields.named.iter().for_each(|field| {
let name = &field.ident.clone().unwrap();
let ty = &field.ty;
quote!{
impl Foo {
pub set_bar(&mut self, v: #ty) {
self.bar = Some(v);
}
}
};
});
}
_ => {}
},
_ => panic!("You can derive it only from struct"),
}
发布于 2019-03-21 17:34:02
你应该像这个未经测试的例子那样做:
use syn::{GenericArgument, PathArguments, Type};
fn extract_type_from_option(ty: &Type) -> Type {
fn path_is_option(path: &Path) -> bool {
leading_colon.is_none()
&& path.segments.len() == 1
&& path.segments.iter().next().unwrap().ident == "Option"
}
match ty {
Type::Path(typepath) if typepath.qself.is_none() && path_is_option(typepath.path) => {
// Get the first segment of the path (there is only one, in fact: "Option"):
let type_params = typepath.path.segments.iter().first().unwrap().arguments;
// It should have only on angle-bracketed param ("<String>"):
let generic_arg = match type_params {
PathArguments::AngleBracketed(params) => params.args.iter().first().unwrap(),
_ => panic!("TODO: error handling"),
};
// This argument must be a type:
match generic_arg {
GenericArgument::Type(ty) => ty,
_ => panic!("TODO: error handling"),
}
}
_ => panic!("TODO: error handling"),
}
}
没有太多需要解释的事情,它只是“展开”了一个类型的不同组成部分:
Type
-> TypePath
-> Path
-> PathSegment
-> PathArguments
-> AngleBracketedGenericArguments
-> GenericArgument
-> Type
.
如果有更简单的方法,我很乐意知道。
请注意,由于syn
是一个解析器,因此它与标记一起工作。您不能确定这是一个Option
。例如,用户可以键入std::option::Option
或编写type MaybeString = std::option::Option<String>;
。您不能处理这些任意的名称。
https://stackoverflow.com/questions/55271857
复制相似问题