자바 상세 분해 류 캐리어 의 부모 위임 및 부모 위임 깨 기

5391 단어
자바 상세 분해 류 캐리어 의 부모 위임 및 부모 위임 깨 기
일반적인 장면 에서 자바 의 기본 클래스 로 더 를 사용 하면 되 지만 어떤 목적 을 달성 하기 위해 서 는 자신의 클래스 로 더 를 실현 해 야 합 니 다. 예 를 들 어 라 이브 러 리 의 상호 격 리 를 달성 하기 위해 서, 예 를 들 어 열 배치 재 로드 기능 을 달성 하기 위해 서 입 니 다.이 때 는 클래스 로 더 를 정의 해 야 합 니 다. 각 클래스 로 더 는 각각의 라 이브 러 리 자원 을 불 러 와 자원 격 리 효 과 를 얻 을 수 있 습 니 다.자원 에 대한 로드 에 있어 부모 위임 체 제 를 계속 사용 할 수도 있 고 부모 위임 체 제 를 깨 뜨 릴 수도 있다.
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();
    }
  }

}


읽 어 주 셔 서 감사합니다. 여러분 에 게 도움 이 되 기 를 바 랍 니 다. 본 사이트 에 대한 여러분 의 지지 에 감 사 드 립 니 다!

좋은 웹페이지 즐겨찾기