ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

注解和反射

2022-05-15 15:35:10  阅读:226  来源: 互联网

标签:aClass 反射 System println 注解 Class out


java注解和反射

注解

注解(Annotation)

  • 作用
    • 可以对程序作出解释(和注释(comment)相同)
    • 可以被其他程序读取(编译器等)
  • 格式:@注释名,还可以添加一些参数值
    • eg:@SuppressWarnings(value="unchecked")
  • 使用范围:package、class、method、field等,可以通过反射机制编程实现对这些元数据的访问

内置注解

  • @Override:重写注解
  • @Deprecated:不鼓励使用的属性、类(通常是危险的或者有更好的选择)
  • @SuppressWarnings:抑制编译时的警告信息(需要添加一个参数才能正常使用)
    • @SuppressWarnings("all")
    • @SuppressWarnings("unchecked")
    • @SuppressWarnings(value={"unchecked","deprecation})

元注解

负责注解其他注解

  • java的4个标准元注解(meta-annotation)
    • @Target:描述注解的使用范围
    • @Retention:表示需要在什么级别保存注解信息,用于描述注解的生命周期(SOURCE(源码时有效)<CLASS(class文件有效)<RUNTIME(运行时有效)(一般定义为RUNTIME))
    • @Documented:该直接将被包含到javadoc中(生成文档注释)
    • @Inherited:子类可以继承父类中的注解

自定义注解

@interface 注解名{定义内容}

每个方法实际上是声明了一个配置参数

方法的名称就是参数的名称

返回值类型就是参数的类型(返回值只能是基本类型Class、String、enum)

可以通过default来声明参数的默认值,没有默认值使用时必须给注解赋值

如果只有一个参数成员,一般参数名为value

注解元素必须要有值

public class Demo01 {
    @Annotation(age=18)
    public void text(){

    }
}
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@interface Annotation{
    String name() default "";
    int age();
}

反射

反射对象

Class类

一个类在内存中只有一个Class对象

一个类被加载后,类的整个结构都会被封装在Class对象中

获得Class对象

Person person = new Person();
//通过对象获得
Class c1 = person.getClass();
//forname获得,需要抛出异常
Class c2 = Class.forName("路径名");
//通过类名获得
Class c3 = Person.class;
//基本内置类型的包装类都有一个Type属性
Class c4 = Integer.TYPE;

类的初始化

  • 类的主动引用(一定会发生类的初始化)

    • main方法
    • new一个类
    • 静态成员和静态方法
    • 使用java.lang.reflect包的方法对类进行反射调用
    • 初始化类时父类没有被初始化则先初始化父类
  • 类的被动引用(不会发生类的初始化)

    • 访问静态域时,只有真正声明这个域的类才会初始化(通过子类引用父类的静态变量不会导致子类初始化)
    • 通过数组定义类的引用,不会触发此类的初始化
    • 引用常量不会触发此类的初始化

类加载器

public static void main(String[] args) throws ClassNotFoundException {
        //获得系统类加载器
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println(systemClassLoader);
    
        //获得系统类加载器的父类加载器(拓展类加载器)
        ClassLoader parent = systemClassLoader.getParent();
        System.out.println(parent);
    
        //获得拓展类加载器的父类加载器(根加载器,无法直接获取)
        ClassLoader parent1 = parent.getParent();
        System.out.println(parent1);
    
        //获得指定类是由那个加载器加载的
        ClassLoader classLoader = Class.forName("路径").getClassLoader();
        System.out.println(classLoader);
    
        //获得类加载器可以加载的路径
        System.out.println(System.getProperty("java.class.path"));
}

类的结构

public static void main(String[] args) throws ClassNotFoundException {
        Class<?> aClass = Class.forName("路径");
    
        //获得类名
        System.out.println(aClass.getName());//类名+包名
        System.out.println(aClass.getSimpleName());//类名
    
        //获得类的属性
        Field[] fields = aClass.getFields();//只能获得public属性
        for (Field field : fields) {
            System.out.println(field);
        }
        Field[] declaredFields = aClass.getDeclaredFields();//能获得所有属性
        for (Field declaredField : declaredFields) {
            System.out.println(declaredField);
        }
    
        //获得类的方法
        Method[] methods = aClass.getMethods();//本类及父类所有的public方法
        for (Method method : methods) {
            System.out.println(method);
        }
        Method[] declaredMethods = aClass.getDeclaredMethods();//本类的所有方法
        for (Method declaredMethod : declaredMethods) {
            System.out.println(declaredMethod);
        }
    
        //获得构造器
        Constructor<?>[] constructors = aClass.getConstructors();//本类及父类所有的public构造器
        for (Constructor<?> constructor : constructors) {
            System.out.println(constructor);
        }
        Constructor<?>[] declaredConstructors = aClass.getDeclaredConstructors();//本类的构造器
        for (Constructor<?> declaredConstructor : declaredConstructors) {
            System.out.println(declaredConstructor);
        }
}

通过反射来创建对象

public class Demo04 {
    public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //获得class对象
        Class<?> aClass = Class.forName("com.mixian.opp.Demo03");
        
        //创建对象
        Object o = aClass.newInstance();
        System.out.println(o);
        
        //通过构造器创建对象
        Constructor<?> declaredConstructor = aClass.getDeclaredConstructor(String.class);
        Object mixian = declaredConstructor.newInstance("mixian");
        System.out.println(mixian);
        
        //通过反射调用方法
        Method setName = aClass.getDeclaredMethod("setName", String.class);
        setName.invoke(o,"mixian");//激活
        
        //通过反射操作属性
        Field name = aClass.getDeclaredField("name");
        name.setAccessible(true);//私有属性不能直接操作,需要关闭程序的安全监测
        name.set(o,"mixian");
    }
}

通过反射获得注解

public class Demo06 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
        Class<?> aClass = Class.forName("com.mixian.opp.Student");
        //通过反射获得注解
        AnnotatedType[] annotatedInterfaces = aClass.getAnnotatedInterfaces();
        for (AnnotatedType annotatedInterface : annotatedInterfaces) {
            System.out.println(annotatedInterface);
        }
        //获得类value的值
        xian annotation = aClass.getAnnotation(xian.class);
        String value = annotation.value();
        System.out.println(value);
        //获得类指定的注解
        Field age = aClass.getDeclaredField("age");
        xian annotation1 = age.getAnnotation(xian.class);
        System.out.println(annotation1.value());
    }
}
@xian("xian")
class Student{
    @xian("xian")
    private String name;
    @xian("xian")
    private int age;
}
@Target({ElementType.TYPE,ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@interface xian{
    String value();
}

标签:aClass,反射,System,println,注解,Class,out
来源: https://www.cnblogs.com/cherish-0/p/16273300.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有