gpt4 book ai didi

java - 为什么我在 Tomcat 8 上创建的后台线程中会收到 ClassNotFoundException?

转载 作者:行者123 更新时间:2023-11-28 22:52:12 25 4
gpt4 key购买 nike

我在使用 Java 8 部署在 Tomcat 9 上的微服务时遇到问题。更好地解释一下,我在 Tomcat 9 上部署了一个 REST 服务 webapp (WAR),该 webapp 有两个部分:

  • 一个是 REST 服务本身,一旦它被调用,接收一些参数并从 MyPackageDTO 创建一个对象(可序列化)类,所以它调用 RabbitMQ 队列并发布对象到队列,这部分工作得很好。
  • 第二部分由 WebListener 整合,它创建一个与 RabbitMQ 上的队列连接的后台线程作为消费者。这部分在抛出 ClassNotFoundException 时失败将第一部分排队的对象 (MyPackageDTO) 出队。

问题是,所有内容都打包在同一个 JAR 中。 WAR 的类打包在位于其 WEB-INF/lib 的 JAR 中,它们未在 WEB-INF/classes 中解包。 (我正在使用 maven“archiveClasses”为真)

所以,看了一会儿之后,我找到了this post它说的是这样的,我认为问题与我在应用程序部署时在 WebListener 中创建的线程使用的类加载器有关。

这有点难以理解,因为所有东西都在同一个 JAR 和同一个 WAR 中,只有一个 Tomcat 服务器(只有一个 JVM)。

是否可以将类加载器设置为新线程?或者如何在与 WebContainer 线程相同的类加载器上创建后台线程?

我回家后会做一些测试,如果我找到这个问题的根源或至少找到解决方法,我会发布它。

非常感谢。

===============

添加信息:

我正在以这种方式创建后台线程:

@WebListener
public class MainWebListener implements ServletContextListener {

private Thread threadQueueListener;
private QueueThreadListener queueListener;

@Override
public void contextDestroyed(ServletContextEvent arg0) {
try {
queueListener.disconnectQueue();
Thread.sleep(QueueThreadListener.QUEUE_DELIVERY_TIMEOUT + 20);
} catch (Throwable tt) {
tt.printStackTrace();
}
}

@Override
public void contextInitialized(ServletContextEvent arg0) {
try {
queueListener = new QueueThreadListener();
threadQueueListener = new Thread(queueListener);
threadQueueListener.setContextClassLoader(Thread.currentThread().getContextClassLoader());
threadQueueListener.start();
} catch (Throwable tt) {
tt.printStackTrace();
}
}
}

我已经添加了设置 ClassLoader 的行,但我遇到了同样的错误。 ClassNotFoundException。

这是线程类:

public class QueueThreadListener implements Runnable {

public static final String QUEUE_NAME = "MQ_TEST_REST_QUEUE";
public static final long QUEUE_DELIVERY_TIMEOUT = 2000;
private boolean toBeContinued = true;
private Connection connection;

@Override
public void run() {
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
try {
connection = factory.newConnection();
Channel channel = connection.createChannel();
QueueingConsumer consumer = new QueueingConsumer(channel);
channel.basicConsume(QUEUE_NAME, true, consumer);
while (toBeContinued) {
try {
QueueingConsumer.Delivery delivery = consumer.nextDelivery(QUEUE_DELIVERY_TIMEOUT);
if (delivery != null) {
InfoDTO to = (InfoDTO)SerializationUtils.deserialize(delivery.getBody());
System.out.println("--> Package received.. [Name: " + to.getName() + " | Age: " + to.getAge()+"]");
}
} catch (Exception ee) {
ee.printStackTrace();
}
}
connection.close();
} catch (TimeoutException t) {
t.printStackTrace();
} catch (IOException x) {
x.printStackTrace();
} catch (ShutdownSignalException e) {
e.printStackTrace();
} catch (ConsumerCancelledException e) {
e.printStackTrace();
}
}

public void disconnectQueue() {
toBeContinued = false;
}
}

名为 disconnectQueue 的方法用于从队列中“优雅地”断开连接。

InfoDTO 只是一个 POJO 但可序列化:

public class InfoDTO implements Serializable {

private static final long serialVersionUID = 5274775183934376052L;
private String name;
private int age;

public InfoDTO(String name, int age) {
super();
this.name = name;
this.age = age;
}

public String getName() { return name; }
public void setName(String name) { this.name = name; }

public int getAge() { return age; }
public void setAge(int age) { this.age = age; }
}

============

这是堆栈跟踪:

org.apache.commons.lang.SerializationException: java.lang.ClassNotFoundException: com.guambo.test.rest.rabbit.dto.InfoDTO
at org.apache.commons.lang.SerializationUtils.deserialize(SerializationUtils.java:165)
at org.apache.commons.lang.SerializationUtils.deserialize(SerializationUtils.java:192)
at com.guambo.test.rest.rabbit.listener.QueueThreadListener.run(QueueThreadListener.java:58)
at java.lang.Thread.run(Thread.java:745)
Caused by: java.lang.ClassNotFoundException: com.guambo.test.rest.rabbit.dto.InfoDTO
at java.net.URLClassLoader.findClass(URLClassLoader.java:381)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
at java.lang.Class.forName0(Native Method)
at java.lang.Class.forName(Class.java:348)
at java.io.ObjectInputStream.resolveClass(ObjectInputStream.java:626)
at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1613)
at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1518)
at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1774)
at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1351)
at java.io.ObjectInputStream.readObject(ObjectInputStream.java:371)
at org.apache.commons.lang.SerializationUtils.deserialize(SerializationUtils.java:162)

提前致谢。

阿尔瓦罗

最佳答案

SerializationUtils 类使用 ObjectInputStream,这要求 SerializationUtils 必须对其反序列化的 InfoDTO 类具有可见性。您必须将 SerializationUtils 类与 InfoDTO 类一起移动到 WAR 中(推荐),或者将 InfoDTD 类移动到与 SerializationUtils 相同的类路径中(不推荐)。

关于java - 为什么我在 Tomcat 8 上创建的后台线程中会收到 ClassNotFoundException?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36727659/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com