注解使用

基础介绍

  • 注解的写法
    @xxx[(一些信息)]
  • 注解放置的位置
    类的上面 属性上面 方法上面 构造方法上面 参数前面
  • 注解的作用
    1.用来充当注释的作用,用作是一个文字的说明 @Deprecated
    2.用来做代码的检测(验证) @Override
    3.可以携带一些信息(内容) 主流用法,比如testNg框架

java中提供出来可用的注解

@Deprecated 用来说明方法是废弃的
@Override 用来做代码检测,检查该方法是否是一个重写方法
@SupperssWarnings(信息)String[] 一个就直接写值,多个加{"","","","",}
unused 变量定时后未被使用
serial 类实现了序列化接口 不添加序列化ID号
rawtypes 集合没有定义泛型
deprecation 方法已废弃
*unchecked 出现了泛型的问题 可以不检测
all 包含以以上所有

注解中可以携带信息 可以不携带信息

信息类型有限制,只能是如下类型
  • 基本数据类型
  • String类型
  • 枚举类型
  • 注解类型
  • 数组类型[] 数组的内部的数据类型,只能是上面的4中类型

描述一个自定义注解

  1. 通过@interface定义一个新的注解类型
  2. 可以描述public abstract的方法 方法要求返回值必须有 返回值类型如上
    3.需要使用到元注解(java自带的)
  • @Target 描述当前的这个注解可以的位置
  • Retention 描述当前的这个注解存在什么作用域中(源代码文件SOURCE、编译后的字节码文件CLASS、加载虚拟机后的内存RUNTIME
  • Inherited 描述当前这个注解是否能被子类对象继承
  • Docement 描述当前这个注解是否能被文档所记录

使用注解实现IOC+DI功能

注解类代码

one - MyAnnotation

@Target({ElementType.FIELD,ElementType.METHOD,ElementType.CONSTRUCTOR})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface MyAnnotation {
    String[] value();  // 将别人放注解上的信息给别人,至于别人用注解携带的信息去干什么,和注解无关
    String test();
}

注解类代码
NotNUll

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface NotNUll {

}

实体类
**Person **

public class Person {

    @MyAnnotation(test = "我是test",value = {"我是name","20","我是sex"})
    @NotNUll
    private String name;

    @NotNUll
    private Integer age;
    @NotNUll
    private String sex;

    @MyAnnotation(test = "我是test",value = {"我是name","20","我是sex"})
    public Person(){}


    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", sex='" + sex + '\'' +
                '}';
    }
}

实现功能能(IOC+DI,并检验是否为空)
MySpring

public class MySpring {

    public Object getBean(String className) {
        Object object = null;

        try {
            Class clazz = Class.forName(className);
            Constructor con = clazz.getConstructor();
            object = con.newInstance();

            Annotation annotation = con.getAnnotation(MyAnnotation.class);

            Class clazzan = annotation.getClass();
            Method methodTest = clazzan.getMethod("test");
            String test = (String) methodTest.invoke(annotation);
            System.out.println(test);
            Method methodValue = clazzan.getMethod("value");
            String[] vatest = (String[]) methodValue.invoke(annotation);
            for (int i = 0; i < vatest.length; i++) {
                System.out.println(vatest[i]);
            }

            Field[] fields = clazz.getDeclaredFields();
            for (int i = 0; i < fields.length; i++) {
                String fieldName = fields[i].getName();
                String firstLetter = fieldName.substring(0, 1).toUpperCase();
                String otherLetter = fieldName.substring(1);
                StringBuilder setMethodName = new StringBuilder("set");
                setMethodName.append(firstLetter);
                setMethodName.append(otherLetter);

                Class fieldType = fields[i].getType();
                Method setMethod = clazz.getMethod(setMethodName.toString(), fieldType);
                //执行找到的set给对应的属性赋值
                setMethod.invoke(object, fieldType.getConstructor(String.class).newInstance(vatest[i]));
            }
            if (checkisNotNUll((Person) object)) {
                return object;
            } else {
                return new String("有参数为空");
            }

        } catch (Exception e) {
            e.printStackTrace();
        }
        return object;
    }


    public boolean checkisNotNUll(Person obj) {
        System.out.println("进来啦。。。。。。。。。");
        Field[] fields = obj.getClass().getDeclaredFields();
        for (int i = 0; i < fields.length; i++) {
            fields[i].setAccessible(true);
            if (fields[i].isAnnotationPresent(NotNUll.class)){
                //获取字段类型
                System.out.println(fields[i].getType());
                try {
                    //获取字段值
                    Object value = fields[i].get(obj);
                    System.out.println("value: " + value);

                    if (value==null){
                        System.err.println("字段[" + fields[i].getName() + "]不能为空");
                        return false;
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }

        }
        System.out.println("没有问题!!! 返回true啦");
        return true;
    }


}

测试类
TestMain

public class TestMain {

    public static void main(String[] args) {

        MySpring mySpring = new MySpring();
        Person person = (Person) mySpring.getBean("myselfannotation.Person");
        System.out.println(person);
    }
}

运行结果:

我是test
我是name
20
我是sex
进来啦。。。。。。。。。
class java.lang.String
value: 我是name
class java.lang.Integer
value: 20
class java.lang.String
value: 我是sex
没有问题!!! 返回true啦
Person{name='我是name', age=20, sex='我是sex'}

总结

1.checkisNotNUll()方法其实有点多余,但是目的只是为了练习注解这个技术,代码中如果为空了不需要该方法就会报错了
2.注解上的信息本质上是没有任何作用的,关键是你看到注解上的信息,准备用来干什么。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容