Java自定义注解和反射的简单使用

注解的作用或者意义是什么?

注解本身没有任何意义,单独的注解就是一种注释,他需要结合其他如反射、插桩等技术才有意义。

Java 注解(Annotation)又称 Java 标注,是 JDK1.5 引入的一种注释机制。是元数据的一种形式,提供有关于程序但不属于程序本身的数据。注解对它们注解的代码的操作没有直接影响。 

元注解

在定义注解时,注解类也能够使用其他的注解声明。对注解类型进行注解的注解类,我们称之为 meta-annotation(元注解)。声明的注解允许作用于哪些节点使用@Target声明;保留级别由@Retention 声明。其中保留级别如下。

·RetentionPolicy.SOURCE:标记的注解仅保留在源级别中,并被编译器忽略。

·RetentionPolicy.CLASS:标记的注解在编译时由编译器保留,但 Java 虚拟机(JVM)会忽略。

·RetentionPolicy.RUNTIME:标记的注解由 JVM 保留,因此运行时环境可以使用它。

SOURCE < CLASS < RUNTIME,即CLASS包含了SOURCE,RUNTIME包含SOURCE、CLASS。

根据注解的保留级别不同,对注解的使用自然存在不同场景。

       

在Android中我们需要设计接口以供使用者调用时,如出现需要对入参进行类型限定,如限定为资源ID、布局ID等类型参数,将参数类型直接给定int即可。然而,我们可以利用Android为我们提供的语法检查注解,来辅助进行更为直接的参数类型检查与提示。

       

同时,我们也可以通过利用@Intdef来定义自己的入参类型检查。

反射

一般情况下,我们使用某个类时必定知道它是什么类,是用来做什么的,并且能够获得此类的引用。于是我们直接对这个类进行实例化,之后使用这个类对象进行操作。

反射则是一开始并不知道我要初始化的类对象是什么,自然也无法使用 new 关键字来创建对象了。这时候,我们使用 JDK 提供的反射 API 进行反射调用。反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;并且能改变它的属性。是Java被视为动态语言的关键。

首先我们自定义一个注解类,来实现Intent跳转传参

//跳转传参注解
//需要反射赋值,使用RUNTIME
@Retention(RetentionPolicy.RUNTIME)
//限定注解范围,field属性、枚举的常量
@Target(ElementType.FIELD)
public @interface Autowired {
    String value() default "mx";
}

第二步,在SecondActivity中,给参数变量加上注解的标记

public class SecondActivity extends AppCompatActivity {
    @Autowired(value = "age")
    private int age;
    @Autowired(value = "name")
    private String name;
    @Autowired(value = "sex")
    private boolean sex;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }
}

第三步,实现注解工具类,利用反射获取class的属性变量和赋值,具体方法看代码注释

public class AutowiredUtils {

    public static void autowired(Activity activity) {
        //反射获取当前class
        Class<? extends Activity> aClass = activity.getClass();
        //获取intent传递的数据
        Intent intent = activity.getIntent();
        Bundle bundle = intent.getExtras();
        if (bundle == null) {
            return;
        }

        //获得此类所有的成员属性
        Field[] declaredFields = aClass.getDeclaredFields();
        //遍历找出有Autowired注解的成员属性
        for (Field field : declaredFields) {
            if (field.isAnnotationPresent(Autowired.class)) {
                //获取注解类实例
                Autowired annotation = field.getAnnotation(Autowired.class);
                //通过注解类实例,获取传入的key
                String key = TextUtils.isEmpty(annotation.value()) ? field.getName() : annotation.value();

                //只有Parcelable数组类型不能直接设置,其他的都可以
                if (bundle.containsKey(key)) {
                    Object obj = bundle.get(key);
                    //获取数组单个元素的类型
                    Class<?> componentType = field.getType().getComponentType();
                    //判断如果当前成员属性是数组,并且类型是Parcelable的子类,则强转为数组
                    if (field.getType().isArray() && Parcelable.class.isAssignableFrom(componentType)) {
                        Object[] objs = (Object[]) obj;

                        //创建对应类型的数组并由objs拷贝
                        Object[] objects = Arrays.copyOf(objs, objs.length, (Class<? extends Object[]>) field.getType());
                        obj = objects;

                    }
                    //开启获取private成员权限
                    field.setAccessible(true);
                    try {
                        //反射赋值
                        field.set(activity, obj);
                    } catch (IllegalAccessException e) {
                        e.printStackTrace();

                    }
                }
            }
        }
    }
}

最后,在onCreat方法中初始化上述工具类,实现在class的RUNTIME时,自动获取getIntent传值和赋值的操作

public class SecondActivity extends AppCompatActivity {
    //自定义findviewbyid注解类,比较简单,可仿照Autowired注解类的实现
    @InjectView(R.id.tv_age)
    TextView tv_age;
    @InjectView(R.id.tv_name)
    TextView tv_name;
    @InjectView(R.id.tv_sex)
    TextView tv_sex;


    @Autowired(value = "age")
    private int age;
    @Autowired(value = "name")
    private String name;
    @Autowired(value = "sex")
    private boolean sex;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        InjectUtils.injectView(this);
        AutowiredUtils.autowired(this);

        tv_age.setText(age + "");
        tv_name.setText(name);
        tv_sex.setText(sex ? "男" : "女");
    }
}

当传值较多时,这样操作大大减少了我们代码的行数,使得程序的阅读性提高很多

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值