总而言之,在我的应用程序不再需要RMI之后,我无法让几个Java RMI的非守护程序线程关闭。这可以防止JVM在main()完成时退出。
我知道导出UnicastRemoteObject
会导致RMI在成功调用UnicastRemoteObject.unexportObject(Object o,boolean force)
之前保持线程处于打开状态。这是一个例子(无需修改即可运行,JVM将正常退出 - 删除对unexportObject的调用,JVM永远不会退出):
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;
import java.rmi.server.UnicastRemoteObject;
public class TestUnicastRemoteObject{
private static UnicastRemoteObject obj;
private static Registry registry;
public static void main(String[] args) throws Exception{
obj = new UnicastRemoteObject(){
private static final long serialVersionUID = 1L;
};
System.err.println("created UnicastRemoteObject");
System.err.println("creating registry ...");
registry = LocateRegistry.createRegistry(9999);
System.err.println("registry created.");
System.err.println("binding obj to registry ...");
registry.bind("Test", obj);
System.err.println("bound");
UnicastRemoteObject.unexportObject(obj, true);
System.err.println("unexported obj");
}
}
此外,您是否创建注册表和/或将远程对象绑定到它似乎并不重要 - 在此示例中唯一似乎重要的是,无论何时创建UnicastRemoteObject,您都需要调用unexportObject
以防止在完成后保留任何线程。
在我的应用程序中,我确保在我创建的每个UnicastRemoteObject上调用了unexportObject,但是RMI的“reaper”线程和“连接接受”线程仍然存在,阻止我的JVM在我的应用程序完成后退出使用RMI资源。
除了忘记取消导出UnicastRemoteObjects之外,是否有其他可能导致RMI留下线程的东西?
答案 0 :(得分:7)
果然,我在代码中遇到了一个错误,导致我的(很多)UnicastRemoteObjects中的一个在调用应用程序完成后不会自动导出。所以答案是:
在正在运行的JVM中取消导出所有UnicastRemoteObjects就足以关闭所有RMI非守护程序线程。
答案 1 :(得分:7)
听起来你解决了问题@Ben但是对于后人,我想我会将我的评论推广到一个答案。每当我有一个注册/取消注册类型的模式时,我都要确保通过单例对象来管理它们。这样你就有一个地方可以找出哪个对象没有注册。在JMX中公开这也是一个胜利。
以下代码之类的东西会很好。它将允许您进行日志记录或JMX查询,以查看哪些对象已绑定到注册表但尚未解除绑定。
public class UnicastRegistry {
private static Registry registry;
private static UnicastRegistry singleton;
// private to force the singleton
private UnicastRegistry() throws RemoteException {
registry = LocateRegistry.createRegistry(9977);
}
public static UnicastRegistry createSingleton() throws RemoteException {
if (singleton == null) {
singleton = new UnicastRegistry();
}
return singleton;
}
public void register(String label, Remote obj) throws Exception {
registry.bind(label, obj);
}
public void unregister(String label) throws Exception {
Remote remote = registry.lookup(label);
registry.unbind(label);
if (remote instanceof UnicastRemoteObject) {
UnicastRemoteObject.unexportObject(remote, true);
}
}
public void unregisterAll() throws Exception {
for (String label : registry.list()) {
unregister(label);
}
}
public void printStillBound() throws Exception {
String[] stillBound = registry.list();
if (stillBound.length > 0) {
System.out.println("Still bound = " + Arrays.toString(stillBound));
}
}
}