I got really stuck with a problem concerning java threads and ProcessBuilder
.
I can´t stop the thread maybe someone could help me to figure out why it doesn´t work. I have two classes one GUI
class and one ThreadWorker
class where the actual thread and ProcessBuilder
are.
GUI
class:
package minimum_gui;
import java.awt.Color;
import java.awt.EventQueue;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JButton;
import java.awt.event.ActionListener;
import java.awt.event.ActionEvent;
public class GUI
{
private JFrame jFrame;
ThreadWorker tw = new ThreadWorker("ThreadName");
public GUI()
{
initialize();
}
private void initialize()
{
jFrame = new JFrame();
jFrame.setTitle("Example GUI");
jFrame.setBounds(100,100,730,330);
jFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
jFrame.setResizable(true);
jFrame.getContentPane().setLayout(null);
JPanel panel = new JPanel();
panel.setBackground(Color.WHITE);
panel.setBounds(10, 11, 694, 281);
jFrame.getContentPane().add(panel);
panel.setLayout(null);
JButton btnStart = new JButton("Start");
btnStart.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent arg0) {
tw.start();
}
});
btnStart.setBounds(28, 36, 89, 23);
panel.add(btnStart);
JButton btnStop = new JButton("Stop");
btnStop.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
tw.stopIt();
}
});
btnStop.setBounds(28, 70, 89, 23);
panel.add(btnStop);
}
public static void main(String[] args) {
EventQueue.invokeLater(new Runnable()
{
public void run()
{
try
{
GUI window = new GUI();
window.jFrame.setVisible(true);
}
catch (Exception e)
{
e.printStackTrace();
}
}
});
}
}
And the ThreadWorker class:
package minimum_gui;
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.Arrays;
public class ThreadWorker implements Runnable
{
Thread thread = null;
String threadName;
public ThreadWorker(String name)
{
//thread.currentThread().setName(name);
this.threadName = name;
}
public synchronized void start()
{
if (thread == null)
{
thread = new Thread(this);
thread.start();
}
}
public synchronized void stopIt()
{
if (thread != null)
{
thread = null;
}
}
public synchronized void interruptIt()
{
if (thread != null)
thread.interrupt();
}
public void run() {
while (thread != null)
{
try {
String[] command = { "CMD", "/C","ping -n 20 google.com"};
ProcessBuilder probuilder = new ProcessBuilder(command);
Process process = probuilder.start();
final long start = System.currentTimeMillis();
// Read out dir output
InputStream is = process.getInputStream();
InputStreamReader isr = new InputStreamReader(is);
BufferedReader br = new BufferedReader(isr);
String line;
System.out.printf("Output of running %s is :\n",
Arrays.toString(command));
try {
while ((line = br.readLine()) != null) {
System.out.println(line);
}
} catch (IOException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
} catch (IOException e2) {
// TODO Auto-generated catch block
e2.printStackTrace();
}
}
}
}
You are doing a command ping with 20 iterations ping -n 20 google.com
Printing the result on these lines:
while ((line = br.readLine()) != null) {
System.out.println(line);
}
Well, this while will only end when the ping is completed, and there is no verification about thread state (null or non-null) in this while.
So if you try to stop the work, on the external while
while (thread != null)
It will only stop when ping finishes, after 20 pings.
To stop immediately you need to check the thread state in the inside while too, like
while ((line = br.readLine()) != null && thread != null) {
System.out.println(line);
}