Java 注解和反射

一、注解

1、内置注解

  • @Override 定义在 java.lang.Override , 只适用于修辞方法 , 表示一个方法声明打算重写超类中的另一个方法声明
  • @Deprecated 定义在java.lang.Deprecated , 用于修辞方法 , 属性 , 类 , 表示不鼓励程序员使用这样的元素
  • @SuppressWarnings 定义在java.lang.SuppressWarnings,用来抑制编译时的警告信息,需要添加一个参数才能正确使用

2、元注解

Java定义了4个标准的meta-annotation类型,他们被用来提供对其他annotation类型作说明

  • @Target : 用于描述注解的使用范围
  • @Retention : 表示需要在什么级别保存该注释信息 , 用于描述注解的生命周期(RUNTIME > CLASS >SOURCE)
  • @Document:说明该注解将被包含在javadoc
  • @Inherited:说明子类可以继承父类中的该注解

3、自定义注解

  • @ interface用来声明一个注解 , 格式 : public @ interface 注解名 { 定义内容 }
  • 其中的每一个方法实际上是声明了一个配置参数
  • 方法的名称就是参数的名称
  • 返回值类型就是参数的类型 ( 返回值只能是基本类型,Class , String , enum ).
  • 可以通过default来声明参数的默认值
  • 如果只有一个参数成员 , 一般参数名为value
  • 注解元素必须要有值 , 我们定义注解元素时 , 经常使用空字符串,0作为默认值
public class CustomAnnotation {
    //注解可以显示赋值,如果没有默认值,就必须给注解赋值
    @MyAnnotation(name = "张三")
    public void test() {
    }
}
@Target(value = {ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation {
    //注解的参数:参数类型+参数名()
    String name() default "";
    int age() default 0;
    int id() default -1;//-1代表不存在
    String[] schools() default {"北京大学","清华大学"};

二、反射

1、反射机制

  • 优点:可以实现动态创建对象和编译,体现出很大的灵活
  • 缺点:对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。这类操作总是慢于直接执行相同的操作

2、Class类

(1)特点​​​​​​​

  • Class 本身也是一个类
  • Class 对象只能由系统建立对象
  • 一个加载的类在 JVM 中只会有一个Class实例
  • 一个Class对象对应的是一个加载到JVM中的一个.class文件
  • 每个类的实例都会记得自己是由哪个 Class 实例所生成
  • 通过Class可以完整地得到一个类中的所有被加载的结构
  • Class类是Reflection的根源,针对任何你想动态加载、运行的类,唯有先获得相应的Class对象

(2)主要方法

  • static ClassforName(String name)   返回指定类名nameClass对象
  • Object newInstance()   调用缺省构造函数,返回Class对象的一个实例
  • getName()   返回此Class对象所表示的实体(类,接口,数组类或void)的名称。
  • Class getSuperClass()   返回当前Class对象的父类的Class对象
  • Class[] getinterfaces()   获取当前Class对象的接口
  • ClassLoader getClassLoader()   返回该类的类加载器
  • Constructor[] getConstructors()   返回一个包含某些Constructor对象的数组
  • Method getMothed(Stringname,Class.. T)   返回一个Method对象
  • Field[] getDeclaredFields()   返回Field对象的一个数组

(3)获取Class类的实例

  • 若已知具体的类,通过类的class属性获取,该方法最为安全可靠,程序性能最高。
Class clazz = Person.class;
  • 已知某个类的实例,调用该实例的getClass()方法获取Class对象
Class clazz = person.getClass();
  • 已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException
Class clazz = Class.forName("demo.Student");
  • 内置基本数据类型可以直接用类名.Type
  • 还可以利用ClassLoader

(4)可以有Class对象的类型

  • class:外部类,成员(成员内部类,静态内部类),局部内部类,匿名内部类。
  • interface:接口
  • []:数组
  • enum:枚举
  • annotation:注解@interface
  • primitive type:基本数据类型
  • void
public class Test {
    public static void main(String[] args) {
        Class c1 = Object.class; //类
        Class c2 = Comparable.class; //接口
        Class c3 = String[].class; //一维数组
        Class c4 = int[][].class; //二维数组
        Class c5 = Override.class; //注解
        Class c6 = ElementType.class; //美剧
        Class c7 = Integer.class; //基本数据类型
        Class c8 = void.class; //void
        Class c9 = Class.class; //class

        int[] a = new int[10]; 
        int[] b = new int[100]; 
        Class c10 = a.getClass(); 
        Class c11 = b.getClass();

        System.out.println(c1);
        System.out.println(c2);
        System.out.println(c3);
        System.out.println(c4);
        System.out.println(c5);
        System.out.println(c6);
        System.out.println(c7);
        System.out.println(c8);
        System.out.println(c9);
        //只要元素类型与维度一样,就是同一个Class
        System.out.println(c11==c10);
    }
}

3、内存分析

(1)类的加载过程

 (2)类的加载与ClassLoader

加载
  • 将class文件字节码内容加载到内存中,并将这些静态数据转换为方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象
链接
  • 验证:确保加载的类信息符合JVM规范,没有安全方面的问题
  • 准备:正式为类变量(static)分配内存并设置类变量默认初始值的阶段,这些内存都将在方法区种进行分配
  • 解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程
初始化
  • 执行类构造器< clinit >()方法的过程。类构造器< clinit >()方法是由编译期自动收集类中所有类变量的赋值动作和静态代码块中的语句合并产生的。
  • 当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化。
  • 虚拟机会保证一个类的< clinit >()方法在多线程环境中被正确加锁和同步
public class ClassLoader {
    public static void main(String[] args) {
        A a = new A();
        System.out.println(A.m);
        }
}
class A {
    static {
        System.out.println("A类静态代码块初始化");
        m = 300;
    }
    static int m = 100;
    public A() {
        System.out.println("A类无参构造初始化");
    }
}

 (3)类初始化

类的主动引用(一定会发生类的初始化)
  • 当虚拟机启动,先初始化main方法所在的类
  • new一个类的对象
  • 调用类的静态成员和静态方法
  • 使用java.lang.reflflect包的方法对类进行反射调用
  • 当初始化一个类,如果其父类没有被初始化,则先会初始化它的父类
类的被动引用(不会发生类的初始化)
  • 当访问一个静态域时,只有真正声明这个域的类才会被初始化。
  • 通过数组定义类引用,不会触发此类的初始化
  • 引用常量不会触发此类的初始化    
public class Test {
    static {
        System.out.println("main类被加载");
    }
    public static void main(String[] args) throws ClassNotFoundException {
//        主动引用
//        Son son=new Son();
//        反射会产生主动引用
//        Class.forName("reflection.java.Son");
//        不会触发初始化
//        System.out.println(Son.b);
//        System.out.println(Son.m);
    }
}
class Father{
    static int b=2;
    static{
        System.out.println("父类被加载");
    }
}
class Son extends Father{
    static int c=3;
    static {
        System.out.println("子类被加载");
    }
        static final int m=1;
}

 (4)类加载器

  • 类加载器作用是把类(class)装载进内存
加载器的分类:
  • 引导类加载器:负责Java平台核心库。无法直接获取
  • 扩展类加载器:负责jre/lib/ext目录下的jar包或-D java.ext.dirs指定目录下的jar包装入工作库
  • 系统类加载器:负责java-classpaht或 -D java.class.paht所指的目录下的类玉jar包装入工作库,是最常用的加载器。
public class ClassLoader {
    public static void main(String[] args) throws ClassNotFoundException {
        //获取系统类的加载器
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println(systemClassLoader);
        //获取系统类加载器的父类加载器-->扩展类加载器   
        ClassLoader parent = systemClassLoader.getParent();
        System.out.println(parent);
        //获取扩展类加载器父类加载器-->根加载器(c/c++)  
        ClassLoader parent1 = parent.getParent();
        System.out.println(parent1);
        //测试当前类的加载器
        ClassLoader classLoader = Class.forName("cn.doris.reflection.Demo09_ClassLoader1").getClassLoader();
        System.out.println(classLoader);
        //测试JDK内置类的加载器
        classLoader = Class.forName("java.lang.Object").getClassLoader();
        System.out.println(classLoader);
        //获得系统类加载器可以加载的路径
        System.out.println(System.getProperty("java.class.path"));
        }
}

4、创建运行时类的对象

通过反射获取运行时类的完整结构 

public class Test {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, NoSuchMethodException {
        Class c1 = Class.forName("REFLECTION.User");
        //获得类名
        System.out.println(c1.getName());
        System.out.println(c1.getSimpleName());
        //获得类的属性
        Field[] fields = c1.getFields();//只能知道public属性
        for(Field field : fields) {
            System.out.println(field);
        }
        Field[] declaredFields = c1.getDeclaredFields();//找到全部的属性
        for(Field field : declaredFields) {
            System.out.println(field);
        }
        //获得指定属性的值
        System.out.println(c1.getDeclaredField("name"));
         System.out.println("-----------------");
        //获得类的方法
        Method[] methods = c1.getMethods();//获得本类及其父类的所有public方法
        for (Method method : methods) {
            System.out.println(method);
        }
        System.out.println("------------------");
        Method[] declaredMethods = c1.getDeclaredMethods();//获取本类所有方法
        for (Method method : declaredMethods) {
            System.out.println(method);
        }
        //获得指定的方法
        System.out.println(c1.getMethod("getName", null));
        System.out.println(c1.getMethod("setName", String.class));
        System.out.println("------------------");
        //获得所有的构造器
        Constructor[] constructors = c1.getConstructors();//获取本类的构造器Public
        for (Constructor constructor : constructors) {
            System.out.println(constructor);
        }
        Constructor[] declaredConstructors = c1.getDeclaredConstructors();//获取本类的所有构造器
        for (Constructor constructor : declaredConstructors) {
            System.out.println("-"+constructor);
        }
        //获得指定的构造器
        System.out.println("指定构造器"+c1.getDeclaredConstructor(String.class,int.class,int.class));
    }
}

5、反射操作泛型

  • ParameterizedType : 表示一种参数化类型,比如Collection
  • GenericArrayType : 表示一种元素类型是参数化类型或者类型变量的数组类型
  • TypeVariable : 是各种类型变量的公共父接口
  • WildcardType : 代表一种通配符类型表达式
public class Test {
    public void test1(Map<String,User> map,List<User> list) {
        System.out.println("test1");
    }
    public Map<String,User> test2(){
        System.out.println("test2");
        return null;
    }
    public static void main(String[] args) throws NoSuchMethodException, SecurityException {
        Method method = Test.class.getDeclaredMethod("test1", Map.class,List.class);//1、获得方法
        Type[] genericParameterTypes = method.getGenericParameterTypes();//2、获得参数类型
            for (Type type : genericParameterTypes) {
                System.out.println("--"+type);
                if(type instanceof ParameterizedType) {
                    Type[] actualTypeArguments = ((ParameterizedType)type).getActualTypeArguments();//3、获得真实的参数类型
                    for (Type type2 : actualTypeArguments) {
                        System.out.println(type2);
                    }
                }
            }
        method = Test.class.getDeclaredMethod("test2", null);    
        Type genericReturnType = method.getGenericReturnType();//获取返回参数类型
        if(genericReturnType instanceof ParameterizedType) {
            Type[] actualTypeArguments = ((ParameterizedType)genericReturnType).getActualTypeArguments();
            for (Type type : actualTypeArguments) {
                System.out.println("--"+type);
            }
        }
    }
}

6、反射操作注解

  • getAnnotations
  • getAnnotation
public class Demo {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
    Class c1=Class.forName("AnnotationReflect.Demo.Student1");
//    通过反射获得注解
    Annotation[] annotations=c1.getAnnotations();
    for (Annotation annotation : annotations) {
        System.out.println(annotation);
    }
//    获得注解value的值
    TableDu tableDu=(TableDu)c1.getAnnotation(TableDu.class);
    String value=tableDu.value();
    System.out.println(value);
//    获得类指定的注解
    Field field=c1.getDeclaredField("name");
    FiledStudent annotation=field.getAnnotation(FiledStudent.class);
    System.out.println(annotation.columName());
    System.out.println(annotation.type());
    System.out.println(annotation.length());
}
}
@TableDu("studentinfo")
class Student1{
    @FiledStudent(columName = "Wid",type="int",length = 10)
    private int id;
    @FiledStudent(columName = "Wname",type="int",length = 10)
    private String  name;
    @FiledStudent(columName = "Wsex",type="int",length = 10)
    private String sex;
    public Student1(int id, String name, String sex) {
        this.id = id;
        this.name = name;
        this.sex = sex;
    }
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getSex() {
        return sex;
    }
    public void setSex(String sex) {
        this.sex = sex;
    }
}
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface FiledStudent{
    String columName();
    String type();
    int length();
}
//类名的注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface TableDu{
    String value();
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值