자바 상세 분해 류 캐리어 의 부모 위임 및 부모 위임 깨 기
일반적인 장면 에서 자바 의 기본 클래스 로 더 를 사용 하면 되 지만 어떤 목적 을 달성 하기 위해 서 는 자신의 클래스 로 더 를 실현 해 야 합 니 다. 예 를 들 어 라 이브 러 리 의 상호 격 리 를 달성 하기 위해 서, 예 를 들 어 열 배치 재 로드 기능 을 달성 하기 위해 서 입 니 다.이 때 는 클래스 로 더 를 정의 해 야 합 니 다. 각 클래스 로 더 는 각각의 라 이브 러 리 자원 을 불 러 와 자원 격 리 효 과 를 얻 을 수 있 습 니 다.자원 에 대한 로드 에 있어 부모 위임 체 제 를 계속 사용 할 수도 있 고 부모 위임 체 제 를 깨 뜨 릴 수도 있다.
1. 부모 위임 체제 에 따라 사용자 정의 클래스 로 더 는 간단 합 니 다. ClassLoader 클래스 를 계승 하고 findClass 방법 을 다시 쓰 면 됩 니 다.다음 예:
① 불 러 올 클래스 Test 를 먼저 정의 합 니 다. 간단 합 니 다. 구축 함수 에서 어떤 종류의 로 더 를 불 러 올 지 출력 합 니 다.
public class Test {
public Test(){
System.out.println(this.getClass().getClassLoader().toString());
}
}
② TestClassLoader 클래스 가 ClassLoader 를 계승 하고 findClass 방법 을 다시 쓰 는 것 을 정의 합 니 다. 이 방법 은 Test. class 바이트 흐름 을 읽 고 부모 클래스 의 defineClass 방법 으로 전달 하 는 것 입 니 다.그리고 사용자 정의 누적 로 더 TestClassLoader 를 통 해 Test. class 를 불 러 올 수 있 습 니 다. 불 러 오기 가 완료 되면 'TestLoader' 를 출력 합 니 다.
public class TestClassLoader extends ClassLoader {
private String name;
public TestClassLoader(ClassLoader parent, String name) {
super(parent);
this.name = name;
}
@Override
public String toString() {
return this.name;
}
@Override
public Class> findClass(String name) {
InputStream is = null;
byte[] data = null;
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try {
is = new FileInputStream(new File("d:/Test.class"));
int c = 0;
while (-1 != (c = is.read())) {
baos.write(c);
}
data = baos.toByteArray();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
is.close();
baos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return this.defineClass(name, data, 0, data.length);
}
public static void main(String[] args) {
TestClassLoader loader = new TestClassLoader(
TestClassLoader.class.getClassLoader(), "TestLoader");
Class clazz;
try {
clazz = loader.loadClass("test.classloader.Test");
Object object = clazz.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}
2. 부모 위임 체 제 를 타파 하려 면 ClassLoader 류 를 계승 해 야 할 뿐만 아니 라 loadClass 와 findClass 방법 도 다시 써 야 한다. 다음 과 같은 예:
① Test 클래스 를 정의 한다.
public class Test {
public Test(){
System.out.println(this.getClass().getClassLoader().toString());
}
}
② ClassLoader 를 계승 하 는 TestClassLoader N 류 를 다시 정의 합 니 다. 이 종 류 는 앞의 TestClassLoader 류 와 비슷 하지만 findClass 방법 을 다시 쓰 는 것 외 에 loadClass 방법 도 다시 썼 습 니 다. 기본 적 인 loadClass 방법 은 부모 위임 체제 의 논 리 를 실현 하 는 것 입 니 다. 즉, 부모 클래스 로 더 를 먼저 불 러 오고 불 러 올 수 없 을 때 만 스스로 불 러 옵 니 다.부모 위임 체 제 를 파괴 하기 위해 서 는 loadClass 방법 을 다시 써 야 합 니 다. 즉, System 류 로 더 에 맡 기 려 고 시도 해 야 로드 에 실 패 했 을 때 자신 이 불 러 올 수 있 습 니 다.그것 은 부모 에 게 우선 적재 기 를 맡 기지 않 았 기 때문에 부모 위임 체 제 를 깨 뜨 렸 다.
public class TestClassLoaderN extends ClassLoader {
private String name;
public TestClassLoaderN(ClassLoader parent, String name) {
super(parent);
this.name = name;
}
@Override
public String toString() {
return this.name;
}
@Override
public Class> loadClass(String name) throws ClassNotFoundException {
Class> clazz = null;
ClassLoader system = getSystemClassLoader();
try {
clazz = system.loadClass(name);
} catch (Exception e) {
// ignore
}
if (clazz != null)
return clazz;
clazz = findClass(name);
return clazz;
}
@Override
public Class> findClass(String name) {
InputStream is = null;
byte[] data = null;
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try {
is = new FileInputStream(new File("d:/Test.class"));
int c = 0;
while (-1 != (c = is.read())) {
baos.write(c);
}
data = baos.toByteArray();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
is.close();
baos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return this.defineClass(name, data, 0, data.length);
}
public static void main(String[] args) {
TestClassLoaderN loader = new TestClassLoaderN(
TestClassLoaderN.class.getClassLoader(), "TestLoaderN");
Class clazz;
try {
clazz = loader.loadClass("test.classloader.Test");
Object object = clazz.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}
읽 어 주 셔 서 감사합니다. 여러분 에 게 도움 이 되 기 를 바 랍 니 다. 본 사이트 에 대한 여러분 의 지지 에 감 사 드 립 니 다!
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
다양한 언어의 JSONJSON은 Javascript 표기법을 사용하여 데이터 구조를 레이아웃하는 데이터 형식입니다. 그러나 Javascript가 코드에서 이러한 구조를 나타낼 수 있는 유일한 언어는 아닙니다. 저는 일반적으로 '객체'{}...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.