栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

注解和反射

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

注解和反射

注解 一、什么是注解

注解是给程序看的东西,可以被程序读取。
不是程序本事,可以对程序做出解释

格式:"@注释名",还可以添加一些参数值
在那里使用:可以在package,class, method, field等上面,相当于给他们添加了额外的辅助信息,我们可以通过反射机制编程实现对这些元素的访问。

二、内置注解
@Override  重写注解 
@Deprecated  不提倡程序员使用
@SuppressWarnings 警告正压
元注解meta-annotation

作用:用来注解其他注解

@Target : 用于描述注解的适用范围,是类还是方法等等
@Retention :表示需要在什么级别保存该注释信息,用于描述注解的生命周期
@document:说明该注解将包含在Javadoc中
@Inherited : 说明子类可以继承父类中的该注解

三、自定义注解

使用@interface自定义注解时,自动继承了java.lang.annotation.Annotation

package annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

//自定义注解
public class Test01 {
    //注解可以显示赋值,如果没有默认值,我们必须给注解赋值

    @MyAnnotation2(age = 14)
    public void test() {
    }

    @MyAnnotation3("是庆哥啊")//这个注解只有一个参数,因此这里可以省略名字,直接写值
    public void test2() {
    }

}

//定义注解的作用范围
@Target({ElementType.TYPE, ElementType.METHOD})

//定义注解的作用时间
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation2 {
    //注解的参数:参数类型+参数名()
    String name() default "";

    int age();

    int id() default -1;//如果默认值为-1代表不存在

    String[] schools() default {"南阳理工学院", "清华大学"};
}

@Target({ElementType.TYPE, ElementType.METHOD})//定义注解的作用范围
@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation3 {
    String value();//当只有一个参数时,命名就用value,这是不成文的规定,在使用注解为参数赋值时,是可以省略value的,用其他名字就不行。

}
反射

反射机制使java语言成了动态语言

一、Java反射机制概述

反射相关的API
java.lang.Class :代表一个类
java.lang.reflect.Method :代表类的方法
java.lang.reflect.Field :代表类的成员变量
java.lang.reflect.Constructor :代表类的构造器

反射的优缺点
优点:可以实现动态创建对象和编译,体现出很大的灵活性。
缺点:对性能有影响。

二、获得反射对象
package reflection;

public class Test2 {
    public static void main(String[] args) throws ClassNotFoundException {
        //通过反射获取类的Class对象
        Class c1 = Class.forName("reflection.User");
        System.out.println(c1);

        //一个类在内存中只有一个Class对象,所以下面c1,c2,c3,c4几个的hashcode相同
        //一个类被加载后,类的整个结构都会别封装在Class对象中
        Class c2 = Class.forName("reflection.User");
        Class c3 = Class.forName("reflection.User");
        Class c4 = Class.forName("reflection.User");
        System.out.println(c1.hashCode());
        System.out.println(c2.hashCode());
        System.out.println(c3.hashCode());
        System.out.println(c4.hashCode());
    }
}


//实体类
class User{

    private String name;
    private int age;
    private int id;

    public User() {
    }

    public User(String name, int age, int id) {
        this.name = name;
        this.age = age;
        this.id = id;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    @Override
    public String toString() {
        return "user{" +
                "name='" + name + ''' +
                ", age=" + age +
                ", id=" + id +
                '}';
    }
}

三、Class类的创建方式有哪些

那些类型可以有Class对象

方式:

package reflection;


public class Test3 {
    public static void main(String[] args) throws ClassNotFoundException {
        Person person = new Student();
        System.out.println("这个人是:" + person.name);
        //方式一:通过对象获得
        Class c1 = person.getClass();
        System.out.println(c1.hashCode());
        //方式二:forName()获得
        Class c2 = Class.forName("reflection.Student");
        System.out.println(c2.hashCode());

        //方式三:通过类名.class获得
        Class c3 = Student.class;
        System.out.println(c3.hashCode());

        //方式四:基本内置类型的包装类都有一个Type属性
        Class c4 = Integer.TYPE;
        System.out.println(c4);

        //获得父类类型
        Class c5 = c1.getSuperclass();
        System.out.println(c5.getName());

    }
}
class Person{
    public String name;

    public Person() {
    }

    public Person(String name) {
        this.name = name;
    }

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

}
class Student extends Person{
    public Student() {
        this.name = "学生";
    }
}
class Teacher extends Person{
    public Teacher(){
        this.name = "老师";
    }
}

四、类加载内存分析

Java内存
堆和方法区是线程共享的,栈是私有的

堆:存放new的对象和数组,可以被所有线程共享,不会存放别的对象引用栈:存放基本变量类型(会包含这个基本类型的具体数值),引用对象的变量(存放这个引用在堆里面的具体地址)方法区:可以被所有线程共享,包含了所有的class和static对象

类的加载

过程:加载 -> 链接 -> 初始化

五、分析类的初始化

什么时候会发生类的初始化

类的主动引用(一定会发生类的初始化):
当虚拟机启动,先初始化main方法所在的类
new一个类的对象
调用类的静态成员(除了final常量)和静态方法
使用java.lang.reflect包的方法对类进行反射调用
当初始化一个类,如果其父类没有被初始化,则先会初始化他的父类

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

package reflection;


public class Test6 {
    static {
        System.out.println("Main类被加载");
    }

    
    public static void main(String[] args) throws ClassNotFoundException {
        //1.主动引用
//        Son son = new Son();//当初始化一个类,如果其父类没有被初始化,则先会初始化他的父类
        //反射也会产生主动引用
//        Class.forName("reflection.Son");

        //2.不会产生类的引用的方法
//        System.out.println(Son.b);//当访问一个静态域时,只有真正声明这个域的类才会被初始化。如:当通过子类引用父类的静态变量,不会导致子类初始化
//        Son[] sons = new Son[5];//通过数组定义类引用,不会触发此类的初始化。这只是开辟空间,并没有使用类
        System.out.println(Son.M);//引用常量不会触发此类的初始化
    }
}
class Father{
    static int b =2;
    static {
        System.out.println("父类被加载");
    }
}
class Son extends Father{
    static {
        System.out.println("子类被加载");
        m = 300;
    }
    static int m = 100;
    static final int M = 1;

}
六、类加载器

类加载器的作用

package reflection;

public class Test7 {
    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(parent);
        //测试当前类是哪个加载器加载的
        ClassLoader classLoader = Class.forName("reflection.Test7").getClassLoader();
        System.out.println(classLoader);
    }
}
七、获取类的运行时结构
package reflection;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class Test8 {

    

    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {

        Class c1 = Class.forName("reflection.User");
        //获取字段
        System.out.println("字段区===========================");
        Field[] fields = c1.getFields();//这方法是只能获取public修饰的字段
        for (Field field : fields) {
            System.out.println(field);
        }
        Field[] declaredFields = c1.getDeclaredFields();//获取声明的所有字段,不管是什么修饰符修饰的
        for (Field declaredField : declaredFields) {
            System.out.println(declaredField);
        }
        Field name = c1.getDeclaredField("name");//指定要那个字段
        System.out.println(name);
        Field id = c1.getDeclaredField("id");
        System.out.println(id);//指定要那个字段,必须是public

        
        //获取方法
        System.out.println("方法区=========================");
        Method[] declaredMethods = c1.getDeclaredMethods();
        for (Method declaredMethod : declaredMethods) {
            System.out.println("declared  " + declaredMethod);
        }
        Method declaredMethod = c1.getDeclaredMethod("getId",null);//第二个参数是被获取方法的参数类型的Class对象
        System.out.println(declaredMethod);

        //获取构造方法
        System.out.println("构造方法区===============");
        Constructor[] declaredConstructors = c1.getDeclaredConstructors();
        for (Constructor declaredConstructor : declaredConstructors) {
            System.out.println(declaredConstructor);
        }
        Constructor declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        for (Constructor constructor : declaredConstructors) {
            System.out.println(constructor.getName());
        }
        //获取父类


        Class superclass = c1.getSuperclass();
        System.out.println(superclass);





    }


}
八、动态的创建对象使用方法


package reflection;


import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class Test9 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //获得Class对象
        Class c1 = Class.forName("reflection.User");
        //构造一个对象
        User user = (User) c1.newInstance();//本质是调用了类的无参构造
        System.out.println(user);
        //通过构造器创建一个对象
        //有参
        Constructor constructor1 = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user1 = (User) constructor1.newInstance("庆哥", 1, 1);
        System.out.println(user1);
        //无参
        Constructor constructor2 = c1.getDeclaredConstructor();
        User user3 = (User) constructor2.newInstance();
        System.out.println(user3);

        //通过反射调用普通方法
        //1.先创建一个该类的对象
        User user4 = (User) c1.newInstance();
        //2.通过反射获取方法
        Method setName = c1.getDeclaredMethod("setName", String.class);
        Method setId = c1.getDeclaredMethod("setId", int.class);
        Method setAge = c1.getDeclaredMethod("setAge", int.class);
        //invoke是调用、激活的意思
        //在invoke方法中传递对象(在哪个对象的方法上做操作),之后是方法的参数
        setName.invoke(user4, "庆哥是个谨慎的人");
        setId.invoke(user4, 100);
        setAge.invoke(user4, 18);
        System.out.println(user4);

        //通过反射操作属性
        User user5 = (User) c1.newInstance();
        Field name = c1.getDeclaredField("name");
        //不能直接操作私有属性,需要关闭程序的安全检测
        name.setAccessible(true);
        name.set(user5, "张迪");
        System.out.println(user5.getName());

    }
}

九、反射操作泛型(了解)

学习视频

十、反射获取注解信息

package reflection;

import org.xml.sax.ext.Attributes2;

import java.lang.annotation.*;
import java.lang.reflect.Field;

public class Test12 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
        Class c1 = Class.forName("reflection.Student2");
        //下面这一行在这个代码里面没用,Tablekuang是一个注解,他有点象类也能通过反射获取Class对象
        Class c2 = Class.forName("reflection.Tablekuang");
        //通过反射获得注解
        Annotation[] annotations = c1.getAnnotations();
        for (Annotation annotation : annotations) {
            System.out.println(annotation);
        }


        //获得注解的value的值
        Tablekuang annotation = c1.getAnnotation(Tablekuang.class);
        String value = annotation.value();
        System.out.println(value);

        //获得类指定的注解,(这里指定的注解是name属性的注解)
        Field name = c1.getDeclaredField("name");
        Fieldkuang annotation1 = name.getAnnotation(Fieldkuang.class);
        System.out.println(annotation1.columnName());
        System.out.println(annotation1.type());
        System.out.println(annotation1.length());

    }
}

@Tablekuang("db_student")
class Student2{
    //注解都对应着数据库中的东西
    @Fieldkuang(columnName = "db_name",type = "varchar",length = 3)
    private String name;
    @Fieldkuang(columnName = "db_id",type = "int",length = 3)
    private int id;
    @Fieldkuang(columnName = "db_age",type = "int",length = 3)
    private int age;

    public Student2() {
    }

    public Student2(String name, int id, int age) {
        this.name = name;
        this.id = id;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public int getAge() {
        return age;
    }

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

    @Override
    public String toString() {
        return "Student2{" +
                "name='" + name + ''' +
                ", id=" + id +
                ", age=" + age +
                '}';
    }
}


//类名的注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Tablekuang{
    String value();
}

//属性的注解

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface Fieldkuang{
    String columnName();
    String type();
    int length();
}


转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/750105.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号