접근 확인은 멤버 속성과 멤버 메소드의 사용이 접근 권한(public, protected, default, private)을 준수하는지 확인하는 것입니다.
간단히 말하면 클래스 멤버(속성 또는 메서드)의 액세스 권한이 비공개인 경우 해당 멤버는 클래스 멤버의 액세스 권한이 있는 경우에만 사용할 수 있습니다. public 인 경우 해당 멤버는 모든 클래스에서 사용할 수 있습니다. 클래스 멤버의 액세스 권한이 default인 경우 해당 멤버는 클래스 멤버의 액세스 권한이 있는 경우에만 사용할 수 있습니다. 보호된 경우 Members 멤버는 동일한 패키지의 클래스와 다른 패키지에 있는 해당 클래스의 하위 클래스에서 사용할 수 있습니다.
클래스 멤버의 접근 권한이 기본인데 해당 멤버를 다른 패키지에서 사용하는 경우, 컴파일할 때 컴파일러가 접근 검사를 수행하고 해당 멤버의 사용이 주어진 접근 권한과 일치하지 않는지 확인합니다. 따라서 오류가 보고됩니다.
예를 들어, 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 클래스로 추상화됩니다.
Field를 사용하여 객체의 멤버 속성에 액세스할 수 있습니다. 컴파일러는 멤버 속성의 액세스 권한을 알지 못하며 런타임에만 이를 알 수 있습니다. 따라서 반사 개체(예: 필드)에 대한 액세스 권한 확인은 가상 머신에만 맡길 수 있습니다.
가상 머신 실행 시, 멤버의 사용이 주어진 접근 권한과 일치하지 않는 것으로 확인되면 다음 코드
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) )
setAccessible(boolean flag)
方法是AccessibleObject类中的一个方法,它是Field、 Method、Constructor的公共父类。当Field、Method或Constructor (三者都是反射对象)分别用于设置字段(set(Object obj, Object value)
)或获取字段(get(Object obj)
)、调用方法(invoke(Object obj, Object... args)
)或创建和初始化类的新实例(newInstance(Object... initargs)
)时,将执行运行时访问检查。
引用自《Java核心技术 第十版》
注意:方法名setAccessible很容易让人产生误解,给人的感觉是设置了成员的可访问性,例如,觉得public修饰的成员是任意类都可以访问的,所以可访问标志是true;觉得private修饰的成员只有本类可以访问,所以可访问标志是false。其实不然,不管是什么访问权限,其可访问标志的值都为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)
해당 메서드는 Field, Method, Constructor의 공통 상위 클래스인 AccessibleObject 클래스 A 메서드에 있습니다. 필드, 메서드 또는 생성자(세 가지 모두 반사 개체)를 사용하여 필드를 설정(set(Object obj, Object value)
)하거나 필드를 가져오는 경우(get(Object obj)</code >) code>) 메서드를 호출하거나(<code>invoke(Object obj, Object... args)
) 클래스의 새 인스턴스를 생성하고 초기화합니다(newInstance(Object... initargs) )
),
이 수행됩니다.
"Java 핵심 기술 10판"에서 인용
🎜🎜🎜Note🎜: 메서드 이름 setAccessible은 오해의 소지가 있습니다. 이는 멤버의 접근성을 설정하는 것처럼 보입니다. 예를 들어 public으로 수정된 멤버는 모든 클래스에서 액세스할 수 있는 것처럼 보입니다. . 이므로 접근 가능 플래그는 true입니다. 이 클래스만 비공개로 수정된 멤버에 액세스할 수 있다고 생각하므로 접근 가능 플래그는 false입니다. 실제로 액세스 권한이 무엇이든 액세스 플래그의 값은 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()) { //屏蔽对象的访问检查 f.setAccessible(true); // 访问不符合访问权限的成员属性 System.out.println(f.getName() + " = " + f.get(p)); } } }
setAccessible(true)
를 전달할 수 있습니다. 오류를 보고하지 않고 개체의 전용 멤버에 액세스할 수 있도록 Java 언어의 (런타임) 액세스 확인을 차단하는 기본값을 수정합니다. 🎜rrreee🎜입력 결과: 🎜🎜🎜🎜위 내용은 Java 리플렉션의 setAccessible() 메소드를 사용하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!