按网上的说法,Rust语言由于没有runtime,因此不支持运行时反射。
我捣鼓了半天发现可以使用声明宏替类添加反射信息,
可以实现类似反射的效果。
贴在这儿供大家参考。
(代码在最底下,只是示意,不完整,请理解原理后自行完善)
简单说明一下这段代码在做什么:
首先代码最后几行就是这个reflect_info!宏的用法:
#[macro_use]
reflect_info! {
    struct Test{
        uuu:u32,
        bbb:bool
    }
}使用cargo expand可以看到这段代码的宏展开结果:

可以看到,Test类经过宏的处理,添加了get_fields(),get_xxx_value(), set_xxx_value() 等方法;
只要用get_fields()获得字段及类型,我们就可以使用基于字段名称透过 getter, setter 来获取类的值或设置类的值。
虽然这个例子只实现了很简单的示意,但大抵还是完成了反射的基本功能;
#[macro_export]
macro_rules! get_u32_value {
    ($name:expr, $condition:expr, $field:expr, u32) => {
        if $name == $condition  {
            return $field;
        }
    };
    ($($else:tt)*) => {};
}
#[macro_export]
macro_rules! get_bool_value {
    ($name:expr, $condition:expr, $field:expr, bool) => {
        if $name == $condition  {
            return $field;
        }
    };
    ($($else:tt)*) => {};
}
#[macro_export]
macro_rules! set_u32_value {
    ($name:expr, $condition:expr, $field:expr, $value:expr, u32) => {
        if $name == $condition  {
            $field = $value;
        }
    };
    ($($else:tt)*) => {};
}
#[macro_export]
macro_rules! set_bool_value {
    ($name:expr, $condition:expr, $field:expr, $value:expr, bool) => {
        if $name == $condition  {
            $field = $value;
        }
    };
    ($($else:tt)*) => {};
}
#[macro_export]
macro_rules! reflect_info {
    (
        $vis1:vis struct $name:ident {
            $(
                $vis2:vis $item:ident: $t:tt$(,)*
            ),+
        }
    ) => {
        $vis1 struct $name {
            $(
                $vis2 $item: $t,
            )+
        }
        impl $name {
            fn get_fields() -> HashMap<&‘static str,&‘static str> {
                let mut fields = HashMap::new();
                $(
                    fields.insert(stringify!($item),stringify!($t));
                )+
                return fields;
            }
            fn get_u32_value(&mut self, name: &str) -> u32{
                $(
                    get_u32_value!(name, stringify!($item), self.$item, $t);
                )+
            }
            fn get_bool_value(&mut self, name: &str) -> u32{
                $(
                    get_bool_value!(name, stringify!($item), self.$item, $t);
                )+
            }
             pub fn set_u32_value(&mut self, name: &str, value: u32){
                $(
                    set_u32_value!(name, stringify!($item), self.$item, value, $t);
                )*
            }
            pub fn set_bool_value(&mut self, name: &str, value: bool){
                $(
                    set_bool_value!(name, stringify!($item), self.$item, value, $t);
                )*
            }
        }
    };
}
#[macro_use]
reflect_info! {
    struct Test{
        uuu:u32,
        bbb:bool
    }
}https://github.com/zkonge/husk/blob/2a75692de441a662af3c1d3dc7dfa2b0c93d73eb/src/tls_item.rs#L52
原文:https://blog.51cto.com/oldycat/2694365