2016-04-03 28 views
0

因此,我有我自己的類加載器類ServerLoader,它能夠從本地服務器加載類,路徑爲http://localhost/app/1/其中存在一些.class文件。但是當我嘗試定義某些類的字節碼時,我的程序崩潰了。線程「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; 
    } 

}  

因此,這裏是主要方法

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 

所有路徑和名稱都是正確的。那麼,什麼會導致這個問題呢?

+0

如果路徑名稱是正確的,你說該文件通過'http://本地主機/應用/ 1/Object.class'存在?否則路徑不正確。 –

+0

什麼是可以工作的類的例子? –

回答

0

我解決了這個問題。 LoadClass方法中的每個ClassLoader都必須首先使用父ClassLoader加載類,然後嘗試以另一種方式加載它。所以這是正確的LoadClass方法。

@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; 
} 

你問到示例類:

package com.server; 

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

    } 
} 
相關問題