java - 将内存中的类日志重定向到 ObjectOutputStream

标签 java class client-server objectoutputstream in-memory

我正在尝试构建一个客户端-服务器 IDE,但在从内存中编译/运行类获取运行时异常的错误日志/异常日志时遇到问题。

我的代码如下所示:

public class ServerThread implements Runnable {

private ObjectInputStream objectInputStream;
private ObjectOutputStream objectOutputStream;
private ByteArrayOutputStream byteArrayOutputStream;
private PrintStream out;

public ServerThread(Socket socket) {
    try {
        objectInputStream = new ObjectInputStream(socket.getInputStream());
        objectOutputStream = new ObjectOutputStream(socket.getOutputStream());
        byteArrayOutputStream = new ByteArrayOutputStream();
        out = new PrintStream(byteArrayOutputStream);
    } catch (IOException ex) {
        ex.printStackTrace();
    }
}

@Override
public void run() {
    try {
        StringBuffer sourceCode = new StringBuffer();
        sourceCode.append(objectInputStream.readObject());

        Class<?> myClass = MyJavaCompiler.compile("Test", sourceCode.toString());
        Method mainMethod = myClass != null ? myClass.getDeclaredMethod("main", String[].class) : null;
        Object myObject = myClass.getConstructor().newInstance();
        try {
            System.setOut(out);
            System.out.print(mainMethod.invoke(myObject, new Object[]{null}));
            objectOutputStream.writeObject(byteArrayOutputStream.toString());
        } catch (InvocationTargetException e) {
            Class<?>[] exceptionsList = mainMethod.getExceptionTypes();
            for(Class<?> c : exceptionsList){
                objectOutputStream.writeObject(c.getName());
                }
            }
        } catch (Exception e) {}
    }
}

这是编译器:

import javax.tools.*;
import java.util.Arrays;

/**
 * Created by alin on 8/23/15.
 */
public class MyJavaCompiler {

private static JavaCompiler javac = ToolProvider.getSystemJavaCompiler();

public static Class<?> compile(String className, String sourceCodeInText) throws Exception {

    JavaSourceFromString sourceCode = new JavaSourceFromString(className, sourceCodeInText);

    CompiledCode compiledCode = new CompiledCode(className);
    Iterable<? extends JavaFileObject> compilationUnits = Arrays.asList(sourceCode);
    DynamicClassLoader dynamicClassLoader = new DynamicClassLoader(ClassLoader.getSystemClassLoader());
    MyJavaFileManager fileManager = new MyJavaFileManager(javac.getStandardFileManager(null, null, null), compiledCode, dynamicClassLoader);

    JavaCompiler.CompilationTask task = javac.getTask(null, fileManager, null, null, null, compilationUnits);
    boolean result = task.call();
    if(result){
        System.out.println("SUCCESS");
        return dynamicClassLoader.loadClass(className);
    }
    else{
        System.out.println("FAIL");
        return null;
        }
    }
}

此时,当我的内存类中出现 NullPointerException 时,我会将“[]”发送到客户端。

编辑:

感谢霍尔格的回答和反馈。我设法通过简单地添加 e.printStackTrace(PrintStream) 来获取代码来执行我想要的操作,代码如下:

try {
            System.setOut(out);
            System.out.print(mainMethod.invoke(myObject, new Object[]{null}));
            objectOutputStream.writeObject(byteArrayOutputStream.toString());
        } catch (InvocationTargetException e) {
            e.printStackTrace(out);
            objectOutputStream.writeObject(byteArrayOutputStream.toString());
        }

这个想法是将我的错误输出从内存中的类重定向到服务器的客户端。

最佳答案

您应该习惯 null 的一致行为治疗。这里

Class<?> myClass = MyJavaCompiler.compile("Test", sourceCode.toString());
Method mainMethod = myClass != null ? myClass.getDeclaredMethod("main", String[].class) : null;
Object myObject = myClass.getConstructor().newInstance();

您正在考虑 myClass可能是null初始化时mainMethod但在下一行中,您将无条件调用 getConstructor()myClass这将导致 NullPointerExceptionmyClassnull 。即使没有失败,您也已分配 nullmainMethod万一myClassnull ,但要打电话invoke ,再次不检查 null .

尽管如此,当您的 compile方法返回一个非 null值,您将继续,但如果您收到 InvocationTargetException ,您不会发送与该异常相关的任何内容,事实上您没有访问 e根本没有,但您发送的是 mainMethod.getExceptionTypes() 的结果该方法通过 throws 声明的类型有哪些?条款。该列表可能确实是空的,main方法不需要声明任何异常,仍然可能抛出 RuntimeExceptionError s。

你也在表演myClass.getConstructor().newInstance(); try … catch 之外子句以及此构造函数抛出的任何 throwable 都将由外部 try … catch 处理以最糟糕的方式,catch (Exception e) {} ...

关于java - 将内存中的类日志重定向到 ObjectOutputStream,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32398224/

相关文章:

java - 从数组中读取 N 个相邻元素的组

java - 如何将 23214+3157x56 等数学表达式中的逗号表示为 23,214+3,157x56?

javascript - 单击子项时需要干净的 jQuery 来更改父类

c# - 在 c++/c# 中从哪里开始抓取/爬行?

java - 服务器向客户端发送信息但无法得到答案

java - java中的字符串操作以获取另一个字符串

java - 如何通过CXF WSDL2JAVA生成代码获取Soap Header信息?

java - 相同的java代码,两个.class文件的类加载问题?

c++ - 类中的结构

authentication - 为什么要使用证书对客户端进行身份验证?