如何在Linux和Windows中正常停止Java进程?
什么时候Runtime.getRuntime().addShutdownHook
被调用,什么时候不被调用?
终结器又如何呢?
我可以从外壳向Java进程发送某种信号吗?
我正在寻找最好的便携式解决方案。
Answers:
在所有未强制终止VM的情况下,都会执行关机挂钩。因此,如果要发出“标准” kill(SIGTERM
通过kill命令),则它们将执行。同样,它们将在调用后执行System.exit(int)
。
但是强行杀死(kill -9
或kill -SIGKILL
),然后它们将不会执行。同样(很明显),如果您从计算机上取下电源,将其放入沸腾的熔岩桶中或用大锤将CPU粉碎成碎片,它们将不会执行。您可能已经知道了。
终结器确实也应该运行,但是最好不要依赖于终结器来进行关闭清理,而要依靠您的关闭钩子来干净地停止运行。而且,一如既往,要小心死锁(我已经看到太多的关闭挂钩会挂起整个过程)!
好的,在我选择了使用“ Java监视和管理”之后,所有的可能性都
在这里概述
,使您可以相对容易地从一个应用程序控制另一个应用程序。您可以从脚本中调用控制应用程序,以在杀死应用程序之前正常停止它。
这是简化的代码:
受控的应用程序:
使用以下VM参数运行它:
-Dcom.sun.management.jmxremote
-Dcom.sun.management.jmxremote.port = 9999
-Dcom.sun.management.jmxremote.authenticate = false
-Dcom.sun.management。 jmxremote.ssl = false
//ThreadMonitorMBean.java
public interface ThreadMonitorMBean
{
String getName();
void start();
void stop();
boolean isRunning();
}
// ThreadMonitor.java
public class ThreadMonitor implements ThreadMonitorMBean
{
private Thread m_thrd = null;
public ThreadMonitor(Thread thrd)
{
m_thrd = thrd;
}
@Override
public String getName()
{
return "JMX Controlled App";
}
@Override
public void start()
{
// TODO: start application here
System.out.println("remote start called");
}
@Override
public void stop()
{
// TODO: stop application here
System.out.println("remote stop called");
m_thrd.interrupt();
}
public boolean isRunning()
{
return Thread.currentThread().isAlive();
}
public static void main(String[] args)
{
try
{
System.out.println("JMX started");
ThreadMonitorMBean monitor = new ThreadMonitor(Thread.currentThread());
MBeanServer server = ManagementFactory.getPlatformMBeanServer();
ObjectName name = new ObjectName("com.example:type=ThreadMonitor");
server.registerMBean(monitor, name);
while(!Thread.interrupted())
{
// loop until interrupted
System.out.println(".");
try
{
Thread.sleep(1000);
}
catch(InterruptedException ex)
{
Thread.currentThread().interrupt();
}
}
}
catch(Exception e)
{
e.printStackTrace();
}
finally
{
// TODO: some final clean up could be here also
System.out.println("JMX stopped");
}
}
}
控制应用程序:
以stop或start作为命令行参数运行它
public class ThreadMonitorConsole
{
public static void main(String[] args)
{
try
{
// connecting to JMX
System.out.println("Connect to JMX service.");
JMXServiceURL url = new JMXServiceURL("service:jmx:rmi:///jndi/rmi://:9999/jmxrmi");
JMXConnector jmxc = JMXConnectorFactory.connect(url, null);
MBeanServerConnection mbsc = jmxc.getMBeanServerConnection();
// Construct proxy for the the MBean object
ObjectName mbeanName = new ObjectName("com.example:type=ThreadMonitor");
ThreadMonitorMBean mbeanProxy = JMX.newMBeanProxy(mbsc, mbeanName, ThreadMonitorMBean.class, true);
System.out.println("Connected to: "+mbeanProxy.getName()+", the app is "+(mbeanProxy.isRunning() ? "" : "not ")+"running");
// parse command line arguments
if(args[0].equalsIgnoreCase("start"))
{
System.out.println("Invoke \"start\" method");
mbeanProxy.start();
}
else if(args[0].equalsIgnoreCase("stop"))
{
System.out.println("Invoke \"stop\" method");
mbeanProxy.stop();
}
// clean up and exit
jmxc.close();
System.out.println("Done.");
}
catch(Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
而已。:-)
另一种方式:您的应用程序可以打开服务器套接字并等待信息到达。例如,一个带有“ magic”字词的字符串:),然后作出反应以关闭:System.exit()。您可以使用诸如telnet之类的外部应用程序将此类信息发送到袜子。
这是一个棘手的但可移植的解决方案:
我实现了Java代理。它可以在Github上找到:https : //github.com/everit-org/javaagent-shutdown
有关解决方案的详细说明,请参见此处: https //everitorg.wordpress.com/2016/06/15/shutting-down-a-jvm-process/
感谢您的回答。关机钩接缝就像适用于我的情况。但是我也碰到了所谓的“监视和管理bean”:
http : //java.sun.com/j2se/1.5.0/docs/guide/management/overview.html
这为远程监视和操作提供了一些不错的可能性Java进程。(在Java 5中引入)