자바 클래스 의 열 교체 사용자 정의 ClassLoader 불 러 오기. class

본 고 는 자바 열 교체 실험 으로 참고 하 였 다.
자바 류 의 열 교체 - 개념, 디자인 과 실현http://www.ibm.com/developerworks/cn/java/j-lo-hotswapcls/index.html
이전 오 류 는 ClassLoader 를 통합 하지 않 고 직접 사용 한 것 입 니 다.
이 문 구 는. jar 를 불 러 오 는 데 만 적 용 됩 니 다.
. class 를 불 러 오 려 면 사용자 정의 ClassLoader 를 사용 해 야 합 니 다.
classloader = (URLClassLoader)ClassLoader.getSystemClassLoader();

GetPI. java 내용:
package yerasel;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.HashSet;

/*
 *       ,   ClassLoader,    .class
 */
class HowswapCL extends ClassLoader {

	private String basedir; //                    
	private HashSet dynaclazns; //                

	public HowswapCL(String basedir, String[] clazns) {
		super(null); //          null
		this.basedir = basedir;
		dynaclazns = new HashSet();
		loadClassByMe(clazns);
	}

	private void loadClassByMe(String[] clazns) {
		for (int i = 0; i < clazns.length; i++) {
			loadDirectly(clazns[i]);
			dynaclazns.add(clazns[i]);
		}
	}

	private Class loadDirectly(String name) {
		Class cls = null;
		StringBuffer sb = new StringBuffer(basedir);
		String classname = name.replace('.', File.separatorChar) + ".class";
		sb.append(File.separator + classname);
		File classF = new File(sb.toString());
		try {
			cls = instantiateClass(name, new FileInputStream(classF),
					classF.length());
		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		return cls;
	}

	private Class instantiateClass(String name, InputStream fin, long len) {
		byte[] raw = new byte[(int) len];
		try {
			fin.read(raw);
			fin.close();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

		return defineClass(name, raw, 0, raw.length);
	}

	protected Class loadClass(String name, boolean resolve)
			throws ClassNotFoundException {
		Class cls = null;
		cls = findLoadedClass(name);
		if (!this.dynaclazns.contains(name) && cls == null)
			cls = getSystemClassLoader().loadClass(name);
		if (cls == null)
			throw new ClassNotFoundException(name);
		if (resolve)
			resolveClass(cls);
		return cls;
	}
}

/*
 *   500ms    ,    class
 */
class Multirun implements Runnable {
	public void run() {
		try {
			while (true) {
				//               
				// class      package       
				String url = System.getProperty("user.dir") + "/lib";// "/lib/yerasel/GetPI.jar";
				HowswapCL cl = new HowswapCL(url,
						new String[] { "yerasel.GetPI" });
				Class cls = cl.loadClass("yerasel.GetPI");
				Object foo = cls.newInstance();
				//         
				Method m = foo.getClass().getMethod("Output", new Class[] {});
				m.invoke(foo, new Object[] {});
				Thread.sleep(500);
			}

		} catch (Exception ex) {
			ex.printStackTrace();
		}
	}
}

public class Test {

	public static Method initAddMethod() {
		try {
			Method add = URLClassLoader.class.getDeclaredMethod("addURL",
					new Class[] { URL.class });
			add.setAccessible(true);
			return add;
		} catch (Exception e) {
			throw new RuntimeException(e);
		}
	}

	public static void main(String[] args) {

		//        
		Thread t;
		t = new Thread(new Multirun());
		t.start();
	}
}

다른 GetPI. java 는 출력 할 때 만 aaaOutput 으로 변경 합 니 다.
jar 패키지 생 성 명령
package yerasel;

import java.io.PrintStream;

public class GetPI
{
  public static double Darts(int n)
  {
    int k = 0;
    double x = 0.0D;
    double y = 0.0D;
    for (int i = 0; i < n; i++)
    {
      x = Math.random();
      y = Math.random();

      if (x * x + y * y <= 1.0D)
        k++;
    }
    return 4 * k / n;
  }
  //        ,   Darts      ,   
  public static void Output() {
    System.out.println("Output");
  }
}

주의해 야 할 것 은: class 는 package 이름 의 폴 더 아래 에 두 어야 합 니 다.
본문 에서. class 는 / lib (사용자 정의 경로) / yerasel (package 이름) 아래 에 놓 여 있 습 니 다.
실행 할 때 프로그램 은 지정 한 폴 더 에 있 는 'class' 를 불 러 옵 니 다. 즉, aaaOutput 을 출력 한 다음 수 동 으로 다른' class' (출력 된 'class') 를 덮어 씁 니 다.
프로그램 즉시 출력 시작
실행 결과 그림:

좋은 웹페이지 즐겨찾기