>Java >java지도 시간 >자바 반사 반사 기본

자바 반사 반사 기본

高洛峰
高洛峰원래의
2016-11-21 10:54:221533검색

1. 반사

반사: JAVA 반사 메커니즘은 모든 클래스에 대해 모든 객체에 대한 이 클래스의 모든 속성과 메서드를 알 수 있습니다. 메소드와 속성, 이렇게 동적으로 얻은 정보와 객체의 메소드를 동적으로 호출하는 기능을 Java 언어의 반영 메커니즘이라고 합니다.

클래스를 분석하려면 먼저 클래스의 바이트코드 파일 객체를 얻어야 합니다. 해부는 Class 클래스의 메소드를 사용하므로 먼저 각 바이트코드 파일에 해당하는 Class 유형 객체를 얻어야 합니다. 리플렉션 작업은 실제로 Class 객체를 통해 획득됩니다.

*a, java.lang.reflect.Field: 클래스 또는 인터페이스의 단일 필드에 대한 정보와 이에 대한 동적 액세스 권한을 제공합니다. 반영된 필드는 클래스(정적) 필드 또는 인스턴스 필드일 수 있습니다. 운영 클래스의 멤버 변수입니다.
*b, java.lang.reflect.Constructor8742468051c85b06f0a0af9e3e506b5c: 작업 클래스의 생성자입니다.
*c, java.lang.reflect.Method: 운영 클래스의 메소드입니다.

Reflection의 기본을 배우기 전에 Person 객체를 인스턴스로 생성하세요

package com.jalja.org.base.relfect;

public class Person {
    private String name;
    int age;
    public String address;

    public Person() {
    }

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

    Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public Person(String name, int age, String address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

    public void show() {
        System.out.println("show");
    }

    public void method(String s) {
        System.out.println("method " + s);
    }

    public String getString(String s, int i) {
        return s + "---" + i;
    }

    private void function() {
        System.out.println("function");
    }

    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + ", address=" + address
                + "]";
    }

}

2. 클래스의 Class 객체 획득

public static void getClassObject() throws ClassNotFoundException{
        //方式一:Object的getClass()方法
        Person person1=new Person();
        Person person2=new Person();
        Class c1=person1.getClass();
        Class c2=person2.getClass();
        System.out.println(person1==person2);//false
        System.out.println(c1==c2);//true 不管JVM内存中有多少个对象,对于字节码文件来说只有一份
        //方式二:数据类型的静态class属性
        Class c3=Person.class;
        System.out.println(c1==c3);//true
        //方式三:Class 类的静态方法 
        //public static Class<?> forName(String className)throws ClassNotFoundException
        Class c4=Class.forName("com.jalja.org.base.relfect.Person");
        System.out.println(c1==c4);//true
    }

3. java. lang.reflect .Constructor: 객체이며 Constructor 클래스를 사용합니다.

1. 생성자 객체 가져오기

//获取Class 对象所表示的类的构造方法
    public static void getConstructorTest() throws  Exception{
        Class c4=Class.forName("com.jalja.org.base.relfect.Person");
        //1、获取Class 对象所表示的类所有公共构造方法
        //public Constructor<?>[] getConstructors() throws SecurityException
        Constructor [] cs=c4.getConstructors();
        //2、获取Class 对象所表示的类所有构造方法
        //public Constructor<?>[] getDeclaredConstructors() throws SecurityException
        Constructor[] cs2 =c4.getDeclaredConstructors();
        //3、获取Class对象所表示类的指定指定公共构造方法,  parameterTypes 参数是 Class 对象的一个数组 ,是指定数据类型的字节码
        //public Constructor<T> getConstructor(Class<?>... parameterTypes);
        Constructor cs3=c4.getConstructor();//获取公共的无参构造方法的Constructor对象
        //获取 该 构造函数  public Person(String name, int age, String address)
        Constructor cs4=c4.getConstructor(String.class,int.class,String.class);
        //4、获取Clss对象所表示类指定的构造范法官 parameterTypes 参数是 Class 对象的一个数组,它按声明顺序标识构造方法的形参类型的字节码。
        //public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes);
        //获取该构造 函数 private Person(String name)  的Constructor对象
        Constructor cs5=c4.getDeclaredConstructor(String.class);
        
    }

2. 생성자 객체를 통해 Class 객체가 나타내는 클래스의 인스턴스를 생성합니다

public static void createObject() throws Exception{
        Class c4=Class.forName("com.jalja.org.base.relfect.Person");
        //使用此 Constructor 对象表示的构造方法来创建该构造方法的声明类的新实例,并用指定的初始化参数初始化该实例
        //public T newInstance(Object... initargs);
        // Person person=new Person()
        Constructor cs3=c4.getConstructor();//获取公共的无参构造方法的Constructor对象
        Object obj=cs3.newInstance();
        
        //Person person=new Person("jalja", 21, "北京");
        Constructor cs4=c4.getConstructor(String.class,int.class,String.class);
        Object obj1=cs4.newInstance("jalja",21,"北京");
        System.out.println(obj1);//Person [name=jalja, age=21, address=北京]
        
        //实例化一个私有的构造函数 private Person(String name)
        //控制java的访问检查
        //public void setAccessible(boolean flag)
        //将此对象的 accessible 标志设置为指示的布尔值。值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。
        //值为 false 则指示反射的对象应该实施 Java 语言访问检查。 
        Constructor cs5=c4.getDeclaredConstructor(String.class);
        cs5.setAccessible(true);
        Object obj2=cs5.newInstance("张三丰");
        System.out.println(obj2);//Person [name=张三丰, age=0, address=null]
    }

4. .lang.reflect.Field

1. Field 객체 가져오기

//获取Class类的Field对象
    public static void getFieldTest() throws Exception{
        Class cs=Class.forName("com.jalja.org.base.relfect.Person");
        //1、public Field[] getFields() throws SecurityException
        //获取Class 对象所表示的类或接口的所有可访问公共(public修饰的)字段
        Field [] fs=cs.getFields(); 
        //2、public Field[] getDeclaredFields() throws SecurityException
        // 获取Class 对象所表示的类或接口所声明的所有字段。包括公共、保护、默认(包)访问和私有字段,但不包括继承的字段
        Field [] fs1=cs.getDeclaredFields(); 
        //3、public Field getField(String name)throws NoSuchFieldException, SecurityException;
        //获取Class 对象所表示的类或接口的指定公共成员(public修饰)字段。name 参数是一个 String,用于指定所需字段的简称
        Field fs2=cs.getField("address");
        //public Field getDeclaredField(String name)  throws NoSuchFieldException,SecurityException
        //获取 Class 对象所表示的类或接口的指定已声明字段。name 参数是一个 String,它指定所需字段的简称
        Field fs3=cs.getDeclaredField("name");
        System.out.println(fs3);
    }

2. Field 객체를 통해 지정된 클래스 속성을 할당합니다

//使用 Field对象
    public static void createVarValue() throws Exception{
        Class cs=Class.forName("com.jalja.org.base.relfect.Person");
        Object obj=cs.getConstructor().newInstance();
        Field addressField=cs.getField("address");
        //public void set(Object obj, Object value);
        //将指定对象变量上此 Field 对象表示的字段设置为指定的新值。如果底层字段的类型为基本类型,则对新值进行自动解包
        //obj - 应该修改其字段的对象  value - 正被修改的 obj 的字段的新值 
        addressField.set(obj, "北京");
        System.out.println(obj); //Person [name=null, age=0, address=北京]
        
        //对非public修饰的变量操作
        Field nameField=cs.getDeclaredField("name");
        //控制java的访问检查
        nameField.setAccessible(true);
        nameField.set(obj, "张三丰");
        System.out.println(obj);//Person [name=张三丰, age=0, address=北京]
    }

5. .lang.reflect.Method

1. Method 객체 가져오기

//获取Method对象
    public static void getMethodTest() throws Exception{
        Class cs=Class.forName("com.jalja.org.base.relfect.Person");
        //1、public Method[] getMethods() throws SecurityException
        //获取Class 对象所表示的类或接口(包括那些由该类或接口声明的以及从超类和超接口继承的那些的类或接口)的公共 member 方法。
        Method [] m1=cs.getMethods();
        //2、public Method[] getDeclaredMethods() throws SecurityException
        //获取Class 对象表示的类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法
        Method [] m2=cs.getDeclaredMethods();
        //3、public Method getMethod(String name, Class<?>... parameterTypes)throws NoSuchMethodException, SecurityException;
        // 获取Class 对象所表示的类或接口的指定公共成员方法。name 参数是一个 String,用于指定所需方法的简称。parameterTypes 参数是按声明顺序标识该方法形参类型的 Class 对象的一个数组
        Method m3=cs.getMethod("show");//无参的方法
        Method m4=cs.getMethod("method",String.class);//带参的方法
        //public Method getDeclaredMethod(String name, Class<?>... parameterTypes)throws NoSuchMethodException,SecurityException
        // Class 对象所表示的类或接口的指定已声明方法。name 参数是一个 String,它指定所需方法的简称,parameterTypes 参数是 Class 对象的一个数组
        Method m5=cs.getDeclaredMethod("function");//无参的方法
        System.out.println(m5);
    }

2. Method 객체를 통해 지정된 클래스의 메소드를 호출합니다

// Method对象的使用
    public static void createMethod() throws Exception{
        Class cs=Class.forName("com.jalja.org.base.relfect.Person");
        Object obj=cs.getConstructor().newInstance();
        Method m3=cs.getMethod("show");//无参的方法
        //public Object invoke(Object obj,Object... args)
        //对带有指定参数的指定对象调用由此 Method 对象表示的底层方法  obj - 从中调用底层方法的对象    args - 用于方法调用的参数 
        m3.invoke(obj);
        //对带参方法的操作
        Method m4=cs.getMethod("method",String.class);//带参的方法
        m4.invoke(obj,"北京");
        //对有返回值得方法操作
        Method m6=cs.getMethod("getString",String.class,int.class);//带参的方法
        Object str=m6.invoke(obj,"北京",200);
        System.out.println(str);
        //对私有无参方法的操作
        Method m5=cs.getDeclaredMethod("function");
        m5.setAccessible(true);
        m5.invoke(obj);
    }


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.