Java Custom Class Loader - PullRequest
       5

Java Custom Class Loader

0 голосов
/ 31 июля 2010

Есть идеи, почему я получаю эту ошибку?(Да, я посмотрел на ошибку и все еще не нашел решения)

Моя ошибка:

Exception in thread "main" java.lang.ClassFormatError: Truncated class file
 at java.lang.ClassLoader.defineClass1(Native Method)
 at java.lang.ClassLoader.defineClass(Unknown Source)
 at java.lang.ClassLoader.defineClass(Unknown Source)
 at org.fellixombc.mysql.util.MysqlClassLoader.findClass(MysqlClassLoader.java:22)
 at org.fellixombc.mysql.util.MysqlClassLoader.loadClass(MysqlClassLoader.java:14)
 at org.fellixombc.mysql.Main.main(Main.java:9)

Файлы:

Main.java

package org.fellixombc.mysql;

import org.fellixombc.mysql.util.MysqlClassLoader;

public class Main {
    public static void main(String[] args) {
        MysqlClassLoader mcl = new MysqlClassLoader();
        try {
            mcl.loadClass("org.fellixombc.mysql.net.Client");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

Client.java:

package org.fellixombc.mysql.net;

public class Client {
    public Client() {
        System.out.println("Hello!");
    }
}

MysqlClassLoder.java:

package org.fellixombc.mysql.util;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

public class MysqlClassLoader extends ClassLoader {
    public MysqlClassLoader() {
        super(MysqlClassLoader.class.getClassLoader());
    }

    @Override
    public Class<?> loadClass(String className) throws ClassNotFoundException {
        return findClass(className);
    }

    @Override
    public Class<?> findClass(String className) throws ClassNotFoundException {
        byte[] b = null;
        try {
            b = loadClassData(className);
            Class c = defineClass(className, b, 0, b.length);
            if(c != null)
                return c;
            return super.findClass(className);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }

    private byte[] loadClassData(String className) throws IOException {
        int size = className.length();
        byte buff[] = new byte[size];

        // Open the file
        FileInputStream fis = new FileInputStream("bin/" + className.replace('.', File.separatorChar) + ".class");
        fis.available();
        fis.read(buff);
        fis.close();

        return buff;
    }
}

Ответы [ 2 ]

3 голосов
/ 31 июля 2010

Да, вы читаете самое большее число байтов, равное количеству символов в имени файла.Вместо этого вам нужно прочитать весь файл.Вот один из методов, использующий readFully, как вы предложили.

File f = new File("bin/" + className.replace('.', File.separatorChar) + ".class");
DataInputStream is = new DataInputStream(new FileInputStream(f));
int len = (int)f.length();
byte[] buff = new byte[len];
is.readFully(buff);
is.close();
return buff;

Поскольку вы не обрабатываете встроенные классы, такие как Object, я думаю, что вам нужно перехватить FileNotFoundException из loadClassData в вашем findClass, затемвызовите super.findClass.Например:

try {
  try {
    b = loadClassData(className);
  }
  catch(FileNotFoundException fnf) {
    return super.findClass(className);
  }
  Class c = defineClass(className, b, 0, b.length);
  if(c != null)
    return c;
  return super.findClass(className);
} catch (IOException e) {
  e.printStackTrace();
}
return null;
1 голос
/ 31 июля 2010

Вы читаете только N байтов (N = длина имени класса) из файла .class в буфер (в loadClassData) перед его возвратом.

Вам необходимо прочитатьсодержимое всего класса перед возвратом буфера для правильного определения класса.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...