アクセス チェックとは、メンバー属性およびメンバー メソッドの使用がアクセス権 (public、protected、default、private) に従っているかどうかをチェックすることです。
ちょっと理論的すぎるのですが、簡単に言うと、クラス メンバー (属性またはメソッド) のアクセス権限がプライベートの場合、そのメンバーは現在のクラスでのみ使用できます。メンバーのアクセス権限が public の場合、そのメンバーはどのクラスでも使用できます。クラスのメンバーのアクセス権限がデフォルトの場合、メンバーは同じパッケージ内のクラスでのみ使用できます。アクセス権限がデフォルトの場合、そのメンバーは同じパッケージ内のクラスでのみ使用できます。クラスのメンバーの権限が保護されている場合、そのメンバーは同じパッケージ内のクラスおよび他のパッケージ内のクラスのサブクラスで使用できます。
クラス メンバーのアクセス許可がデフォルトであるにもかかわらず、そのメンバーを別のパッケージで使用する場合、コンパイル時に、コンパイラはアクセス チェックを実行しますそして、そのメンバーの使用を検出します。は指定されたアクセス権と矛盾するため、エラーが報告されます。
たとえば、com.example パッケージの下に 4 つのメンバー変数を持つ People クラスを作成します。 com.example.app パッケージ (com.example とは別のパッケージ) で、People クラスの 4 つのメンバー変数を使用します。
package com.example; package com.example; public class People { private int privateVar = 1; int defaultVar = 2; protected int protectedVar = 3; public int publicVar = 4; } package com.example.app; import com.example.People; public class TestMain { public static void main(String[] args) { People p = new People(); System.out.println(p.privateVar); System.out.println(p.defaultVar); System.out.println(p.protectedVar); System.out.println(p.publicVar); } }
コンパイル後、publicVar の使用がパブリック アクセス権に準拠していることを確認するメッセージが表示されるため、エラーはありません。
アクセス チェックが何であるかを皆さんは理解していると思いますが、反射オブジェクトのアクセス チェックとは何でしょうか?
クラスのメンバー属性、メンバー メソッド、およびコンストラクターは、リフレクション中にそれぞれ Field、Method、および Counstructor クラスに抽象化されます。
フィールドを使用して、オブジェクトのメンバー属性にアクセスできます。コンパイラはメンバー属性のアクセス許可を認識せず、実行時にのみ認識します。したがって、反射オブジェクト (フィールドなど) へのアクセス権のチェックは仮想マシンにのみ任せることができます。
仮想マシンの実行中に、メンバーの使用が指定されたアクセス権と矛盾していることが判明した場合、次のコード
package com.example.app; import com.example.People; import java.lang.reflect.Field; public class TestMain { public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException { People p = new People(); Class cl = p.getClass(); // 利用反射访问private修饰的成员变量 Field f = cl.getDeclaredField("privateVar") System.out.println(f.get(p)); } }
実行時に例外がスローされます。 java.lang.IllegalAccessException
要約すると、アクセス チェックはコンパイル時にコンパイラによって実行することも、実行時に仮想マシンによって実行することもできます (主にリフレクション目的)
#setAccessible(boolean flag) メソッドは、Field の共通の親クラスである AccessibleObject クラスのメソッドです。メソッドとコンストラクター。フィールド、メソッド、またはコンストラクター (3 つすべてがリフレクション オブジェクト) が、それぞれフィールドの設定 (set(Object obj, Object value)
) またはフィールドの取得 (
get(Object obj)) に使用される場合) )、メソッドを呼び出すとき (
invoke(Object obj, Object... args))、またはクラスの新しいインスタンスを作成して初期化するとき (
newInstance(Object... initargs)##) #)、実行時アクセス チェック
を実行します。 「Java Core Technology Tenth Edition」より引用
: メソッド名 setAccessible は誤解を招きやすいです。たとえば、パブリックで変更されたメンバーにはどのクラスからもアクセスできると思われる場合、アクセシビリティ フラグは true になります。プライベートで変更されたメンバーは、このクラスのみからアクセスできると思われる場合、アクセシビリティ フラグは間違い。実際、アクセス権が何であっても、アクセス フラグの値は false です。 テスト コードは次のとおりです:
package com.example.app; import com.example.People; import java.lang.reflect.Field; public class TestMain { public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException { People p = new People(); Class cl = p.getClass(); // 打印输出所有成员变量的名字及可访问标志 for (Field f : cl.getDeclaredFields()) { System.out.println(f.getName() + ": " + f.isAccessible()); } } }
入力結果:
上記の API により、これが
であることが非常に明確になります。アクセス可能です フラグは Java 言語アクセス チェックをブロックするかどうかを示します、デフォルト値は false、(上でテスト済み) デフォルト値は
setAccessible によって変更できます(true) の場合、Java 言語の (実行時) アクセス チェックがブロックされるため、エラーを報告せずにオブジェクトのプライベート メンバーにアクセスできます。 <pre class="brush:java;">package com.example.app;
import com.example.People;
import java.lang.reflect.Field;
public class TestMain {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
People p = new People();
Class cl = p.getClass();
for (Field f : cl.getDeclaredFields()) {
//屏蔽对象的访问检查
f.setAccessible(true);
// 访问不符合访问权限的成员属性
System.out.println(f.getName() + " = " + f.get(p));
}
}
}</pre>
入力結果:
以上がJavaリフレクションのsetAccessible()メソッドの使用方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。