2016-04-03 8 views
0

私は独自のクラスローダークラスServerLoaderを持っています。パスはローカルサーバからクラスhttp://localhost/app/1/をロードできます。クラスファイルはいくつか存在します。しかし、いくつかのクラスのバイトコードを定義しようとすると、私のプログラムがクラッシュします。スレッド "main"の例外java.lang.NoClassDefFoundError:java/lang/Object

result = defineClass(name, ByteClass, 0, ByteClass.length); 

だから私はここでException in thread "main" java.lang.NoClassDefFoundError: java/lang/Object

を取得しています全体のクラスです。だからここ

package com.local; 

import java.io.*; 
import java.net.HttpURLConnection; 
import java.net.URL; 
import java.security.MessageDigest; 
import java.security.NoSuchAlgorithmException; 
import java.util.HashMap; 
import java.util.Map; 

public class ServerLoader extends ClassLoader { 

    private ClassLoader parent; 
    private String urlPath; 

    public ServerLoader(String urlPath) { 
     parent = ServerLoader.class.getClassLoader(); 
     this.urlPath = urlPath; 
    } 

    @Override 
    public Class<?> loadClass(String name) throws ClassNotFoundException { 

     Class<?> result = null; 

     try { 
      URL mainUrl = new URL(urlPath + name.substring(11); 
      HttpURLConnection conn = (HttpURLConnection) mainUrl.openConnection(); 
      conn.setRequestMethod("GET"); 

      ByteArrayOutputStream out = new ByteArrayOutputStream(); 
      BufferedInputStream input = new BufferedInputStream(conn.getInputStream()); 

      int data; 
      while ((data = input.read()) != -1) { 
       out.write(data); 
      } 

      byte[] ByteClass = out.toByteArray(); 
      result = defineClass(name, ByteClass, 0, ByteClass.length); 
      cacheClass.put(result.getName(), result); 
      return result; 
     } catch (IOException e) { 
      e.printStackTrace(); 
     } 

     return null; 
    } 

}  

は、mainメソッド

package com.local; 

import java.lang.reflect.Method; 

public class Main { 

    public static void main(String[] args) throws Exception { 
     ServerLoader loader = new ServerLoader("http://localhost/app/1/"); 

     Class<?> remoteInstanceClass = loader.loadClass("com.server.PluginA"); 

     Method mainMethod = remoteInstanceClass.getMethod("main", String[].class); 
     mainMethod.invoke(remoteInstanceClass, (Object) args); 
    } 
} 

とスタックトレース

com.local.Main 
    Connected to the target VM, address: '127.0.0.1:54553', transport: 'socket' 
    java.io.FileNotFoundException: http://localhost/app/1/Object.class 
     at sun.net.www.protocol.http.HttpURLConnection.getInputStream0(HttpURLConnection.java:1836) 
     at sun.net.www.protocol.http.HttpURLConnection.getInputStream(HttpURLConnection.java:1441) 
     at com.local.ServerLoader.ConnectAndResolveClass(ServerLoader.java:43) 
     at com.local.ServerLoader.loadClass(ServerLoader.java:28) 
     at java.lang.ClassLoader.defineClass1(Native Method) 
     at java.lang.ClassLoader.defineClass(ClassLoader.java:763) 
     at java.lang.ClassLoader.defineClass(ClassLoader.java:642) 
     at com.local.ServerLoader.ConnectAndResolveClass(ServerLoader.java:51) 
     at com.local.ServerLoader.loadClass(ServerLoader.java:28) 
     at com.local.Main.main(Main.java:10) 
    Exception in thread "main" java.lang.NoClassDefFoundError: java/lang/Object 
     at java.lang.ClassLoader.defineClass1(Native Method) 
     at java.lang.ClassLoader.defineClass(ClassLoader.java:763) 
     at java.lang.ClassLoader.defineClass(ClassLoader.java:642) 
     at com.local.ServerLoader.ConnectAndResolveClass(ServerLoader.java:51) 
     at com.local.ServerLoader.loadClass(ServerLoader.java:28) 
     at com.local.Main.main(Main.java:10) 
    Disconnected from the target VM, address: '127.0.0.1:54553', transport: 'socket' 

Process finished with exit code 1 
です

すべてのpathesと名前は正しいです。だから何が問題を引き起こす可能性がありますか?

+0

@Override public Class<?> loadClass(String name) throws ClassNotFoundException { Class<?> result = null; try { result = parent.loadClass(name); } catch (ClassNotFoundException e) { try { URL mainURL = new URL(urlPath + name.substring(name.lastIndexOf('.') + 1) + ".class"); HttpURLConnection conn = (HttpURLConnection) mainURL.openConnection(); conn.setRequestMethod("GET"); ByteArrayOutputStream out = new ByteArrayOutputStream(); BufferedInputStream input = new BufferedInputStream(conn.getInputStream()); int data; while ((data = input.read()) != -1) { out.write(data); } byte[] byteClass = out.toByteArray(); result = defineClass(name, byteClass, 0, byteClass.length); } catch (IOException e1) { e1.printStackTrace(); } } return result; } 

は、次の例のクラスについての質問しましたか?それ以外の場合、パスは正しくありません。 –

+0

動作するクラスの例を教えてください。 –

答えて

0

私はこの問題を解決しました。 LoadClassの各ClassLoaderメソッドでは、最初に親ClassLoaderを使用してクラスをロードし、別の方法でロードする必要があります。これは正しいLoadClassメソッドです。 // localhostの/アプリ/ 1/Object.class`が存在する:パス名が正しい場合は、ファイルは `HTTPのことを言っている

package com.server; 

public class PluginE { 
    public static void main(String[] args) { 
     System.out.println("Executing main method of PluginE"); 

    } 
} 
関連する問題