ホームページ  >  記事  >  Javaのリフレクションメカニズムの原理は何ですか

Javaのリフレクションメカニズムの原理は何ですか

DDD
DDDオリジナル
2023-06-14 13:53:5610450ブラウズ

Java リフレクション メカニズムの原則: 私たちが記述するソース コードは .java ファイルであり、javac によってコンパイルされると、.class ファイル、つまりバイトコード ファイルになります。プログラムが実行されると、JVM がバイトコード ファイルをクラス メモリにロードします。厳密に言えば、メソッド領域にロードされ、java.lang.Class オブジェクトに変換されます。リフレクションを通じて、インスタンスを構築し、メンバ変数の値を取得し、メソッドを指定して呼び出します。

Javaのリフレクションメカニズムの原理は何ですか

#この記事の動作環境: Windows 10 システム、java17 バージョン、dell g3 コンピューター。

リフレクションの原理は次のとおりです:

私たちが記述するソース コードは .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);

リフレクションによる定義されたアノテーションの値の取得

1) メンバー変数のアノテーションと値を取得します。

@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();

概要:

リフレクションにより、オブジェクトのメンバ変数の値を取得したり、メソッドを呼び出したり、メンバ変数やメソッドを取得したりできます。オブジェクトに定義されたメソッドとパラメータの注釈。レトロフィットでは、アノテーションとリフレクション技術、および動的プロキシを使用します。

以上がJavaのリフレクションメカニズムの原理は何ですかの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。