Java 리플렉션 메커니즘의 원리: 우리가 작성하는 소스 코드는 javac로 컴파일되어 .class 파일, 즉 바이트코드 파일이 되는 .java 파일입니다. 프로그램이 실행될 때 JVM은 클래스를 로드합니다. 엄격한 의미에서는 메소드 영역에 로드되어 java.lang.Class 객체로 변환된다고 합니다. 인스턴스를 구성하고, 멤버 변수의 값을 가져오고, 메소드를 호출하고 호출하세요.
이 문서의 운영 환경: Windows 10 시스템, java17 버전, dell g3 컴퓨터.
Reflection의 원리는 다음과 같습니다.
저희가 작성하는 소스코드는 .java 파일인데, javac로 컴파일하면 .class 파일, 즉 바이트코드 파일이 됩니다.
프로그램이 실행되면 JVM은 바이트코드 파일을 메모리에 로드합니다. 엄밀히 말하면 메소드 영역에 로드되어 java.lang.Class 객체로 변환됩니다. 모든 것은 객체입니다. 클래스는 클래스에 정의된 속성, 메소드, 생성자, 주석, 인터페이스 및 기타 정보를 포함하여 메타데이터 공간에서 클래스의 데이터 구조를 설명하는 클래스 객체라고 합니다.
리플렉션을 통해 인스턴스를 구성하고, 멤버 변수의 값을 가져오고, 메서드를 가져오고 호출할 수 있습니다. 멤버 변수, 메서드 및 메서드 매개 변수에 정의된 주석을 가져올 수도 있습니다.
구현 코드:
1) 매개변수 없는 인스턴스 구성: 리플렉션을 통해 매개변수 없는 생성자 호출
//1.通过全类名加载字节码对象 Class clazz = Class.forName("com.example.lib.Person"); //2.通过类的字节码拿到定义的构造函数 Constructor constructor = clazz.getConstructor(); //3.通过构造方法创建对象 Object obj = constructor.newInstance();
2) 매개변수화된 인스턴스 구성:
//1.通过全类名加载字节码对象 Class clazz = Class.forName("com.example.lib.Person"); //2.通过类的字节码拿到定义的构造函数 Constructor constructor = clazz.getConstructor(int.class,String.class); //3.通过构造方法创建对象 Object obj = constructor.newInstance(20,"xiaohua");
3) 리플렉션을 통해 멤버 변수의 값을 가져옵니다.
//4.通过属性名获取属性 Field field = clazz.getDeclaredField("age"); field.setAccessible(true); //5.调用get方法拿到对象obj属性age的值 Integer age = (Integer) field.get(obj);
4) 리플렉션을 통해 객체의 메소드를 호출합니다.
//4.通过方法名和参数类型,拿到方法 method = clazz.getMethod("setAge", int.class); //5.调用方法 obj是哪个对象身上的方法。 method.invoke(obj, 21); method = clazz.getMethod("getAge"); method.invoke(obj);
5) 리플렉션을 통해 정적 변수의 값을 가져옵니다.
//1.通过全类名加载字节码对象 Class clazz = Class.forName("com.example.lib.Person"); //2.获取静态属性ID Field field = clazz.getField("ID"); field.setAccessible(true); //3.拿到静态属性ID的值。 // 因为静态变量存在方法区,在对象创建之前,就已经加装到了内存 //所以,没有对象,也可以获取变量的值,这里传null也是可以的。 int id = (int) field.get(null);
reflection을 통해 정의된 Annotation의 값을 가져옵니다
1) Annotation과 멤버변수의 값을 가져옵니다.
@Target({ElementType.FIELD}) @Retention(RetentionPolicy.RUNTIME) public @interface BindView { int value(); }
public class MainActivity { @BindView(10000) TextView textView; }
//10通过反射拿到定义在属性上的注解 Class clazz = MainActivity.class; Field textView = clazz.getDeclaredField("textView"); BindView bindView = textView.getAnnotation(BindView.class); int txtId = bindView.value();
3) 리플렉션을 통해 메소드 및 메소드 매개변수에 정의된 주석 및 값을 가져옵니다.
@Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface Post { String value() default ""; }
public interface NetWorkInterface { @Post("http://www.baidu.com") Call getPerson(@Queue("name") String name, @Queue("200") int price); }
//11通过反射拿到方法上定义的注解 clazz = NetWorkInterface.class; Method method = clazz.getMethod("getPerson", String.class, int.class); //获取Post注解 Post post = method.getAnnotation(Post.class); //获取值 String url = post.value();
//12通过反射拿到参数上的注解 //为是个二维数组,因为方法参数会有多个,一个参数有可能定义多个注解 Annotation[][] annotations = method.getParameterAnnotations(); for (Annotation[] ans : annotations) { for (Annotation an : ans) { if (an instanceof Queue) { Queue queue = (Queue) an; String value = queue.value(); } } }
4) 메소드의 매개변수 및 반환 값 유형을 가져옵니다.
//13.拿到方法参数的类型。 Type[] types = method.getGenericParameterTypes(); for (Type type : types) { System.out.println(type.toString()); } //14.获取方法返回值类型 Type type = method.getGenericReturnType();
요약:
리플렉션을 통해 객체의 멤버 변수 값을 가져오고, 메서드를 호출하고, 멤버 변수, 메서드, 메서드 매개변수에 정의된 주석을 가져올 수 있습니다. Retrofit은 주석과 함께 반사 기술 및 동적 프록시를 사용합니다.
위 내용은 Java 반사 메커니즘의 원리는 무엇입니까의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!