Java 为什么我的相框没有处理?
我正在使用以下方法显示一个帧:Java 为什么我的相框没有处理?,java,swing,Java,Swing,我正在使用以下方法显示一个帧: public static void createImage(final String url, final String file, final String filetype) { UIUtils.setPreferredLookAndFeel(); NativeInterface.open(); SwingUtilities.invokeLater(new Runnable() { @
public static void createImage(final String url, final String file, final String filetype) {
UIUtils.setPreferredLookAndFeel();
NativeInterface.open();
SwingUtilities.invokeLater(new Runnable() {
@SuppressWarnings("deprecation")
@Override
public void run() {
frame = new JFrame();
WebsiteThumbnailCreator ex = new WebsiteThumbnailCreator(url, file, filetype);
frame.getContentPane().add(ex, BorderLayout.CENTER);
frame.setSize(FRAMESIZE);
frame.setLocationByPlatform(true);
frame.setVisible(true);
frame.hide();
}
});
NativeInterface.runEventPump();
}
然后我进行一些计算,然后,我想再次处理帧:
try {
ImageIO.write(rendered, filetype, new File(file + "." + filetype));
frame.dispose();
logger.trace("Tried to dispose the frame");
} catch (IOException e) {
logger.fatal(e.getMessage());
} finally {
logger.debug("Try to dispose the frame");
frame.dispose();
}
我可以看到日志消息,但VM仍在运行。我也想终止它。我做错了什么
似乎仍在运行一个非执事线程,但我不明白为什么:
您没有设置JFrame的默认关闭操作,而是默认为JFrame.HIDE\u ON\u close 添加到JFrame设置代码:
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
如果JFrame是应用程序,并且希望在JFrame关闭时退出所有内容,或者
frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
如果JFrame不是整个应用程序,并且希望停止Swing线程,但继续处理
欲了解更多信息,请查看
编辑:
关于你的评论,
@HoverCraftfullOfels好吧,如果你说的是对的,那么为什么我的框架会在代码中消失:public static void frameTest抛出InterruptedException{final JFrame jf=new JFrameHello World;jf.setbounds100100400400;jf.setVisibletrue;Thread.sleep8000;SwingUtilities.invokeLaternew Runnable{public void run{//TODO自动生成的方法存根jf.dispose;};}
我将其翻译为:
public static void frameTest() throws InterruptedException {
final JFrame jf = new JFrame("Hello World");
jf.setBounds(100, 100, 400, 400);
jf.setVisible(true);
Thread.sleep(8000);
SwingUtilities.invokeLater(new Runnable() {
public void run() {
jf.dispose();
}
});
}
只有当没有更多的非守护进程线程时,才会退出。如果您给它一个非守护进程线程,它将继续运行。例如
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
for (int i = 0; i < 30; i++) {
System.out.println("i := " + i);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
try {
frameTest();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
在本例中,您的代码将继续运行。您没有设置JFrame的默认关闭操作,而是默认为JFrame.HIDE\u ON\u close 添加到JFrame设置代码:
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
如果JFrame是应用程序,并且希望在JFrame关闭时退出所有内容,或者
frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
如果JFrame不是整个应用程序,并且希望停止Swing线程,但继续处理
欲了解更多信息,请查看
编辑:
关于你的评论,
@HoverCraftfullOfels好吧,如果你说的是对的,那么为什么我的框架会在代码中消失:public static void frameTest抛出InterruptedException{final JFrame jf=new JFrameHello World;jf.setbounds100100400400;jf.setVisibletrue;Thread.sleep8000;SwingUtilities.invokeLaternew Runnable{public void run{//TODO自动生成的方法存根jf.dispose;};}
我将其翻译为:
public static void frameTest() throws InterruptedException {
final JFrame jf = new JFrame("Hello World");
jf.setBounds(100, 100, 400, 400);
jf.setVisible(true);
Thread.sleep(8000);
SwingUtilities.invokeLater(new Runnable() {
public void run() {
jf.dispose();
}
});
}
只有当没有更多的非守护进程线程时,才会退出。如果您给它一个非守护进程线程,它将继续运行。例如
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
for (int i = 0; i < 30; i++) {
System.out.println("i := " + i);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
try {
frameTest();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
在此示例中,您的代码将继续运行。只要存在非守护进程线程,VM将继续运行。您可以使用显式退出VM。查看正在运行的线程的最简单方法可能是使用JConsole连接到进程。主线程将继续运行,直到您从主方法i返回n主类。只要存在非守护进程线程,VM将继续运行。您可以使用显式退出VM。查看正在运行的线程的最简单方法可能是使用JConsole连接到进程。主线程将继续运行,直到您从主类中的主方法返回。如果我要打电话来
try {
ImageIO.write(rendered, filetype, new File(file + "." + filetype));
frame.dispose();
logger.trace("Tried to dispose the frame");
} catch (IOException e) {
logger.fatal(e.getMessage());
} finally {
logger.debug("Try to dispose the frame");
frame.dispose();
}
然后代码将被执行,但是
1仅当在EDT上完成时,否则…EDT外部顶层容器将粘贴在屏幕上
或
2个顶级容器错过了方法finalize,然后将永远不会根据Windows NT/2000中的资源对其进行GC限制,这些资源处于活动状态…,然后控制台的输出是正确的,并保持重新绑定不变,直到存在当前JVM实例为止
3仅创建一个JFrame,然后通过重新使用此容器执行另一个用户操作
JFramegetContentPane.removeAll
JFrame.addwhich
Java7 JFramerevalidate;对于Java6,最好使用JPanel或ContentPane中的容器也可以这样做
JFramerepaint
编辑
您可以测试的DefaultCloseOperations和JFramedispose的其余设置
您在这里发布的代码与此代码中来自JMenuItem的代码相同
import javax.swing.*;
import java.awt.*;
import java.awt.event.*;
import javax.swing.border.EmptyBorder;
public class ClosingFrame extends JFrame {
private JMenuBar MenuBar = new JMenuBar();
private JFrame frame = new JFrame();
private static final long serialVersionUID = 1L;
private JMenu File = new JMenu("File");
private JMenuItem Exit = new JMenuItem("Exit");
private JFrame frame1 = new JFrame();
public ClosingFrame() {
File.add(Exit);
MenuBar.add(File);
Exit.setBorder(new EmptyBorder(10, 10, 10, 10));
Exit.addActionListener(new ExitListener());
WindowListener exitListener = new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {
int confirm = JOptionPane.showOptionDialog(frame,
"Are You Sure to Close this Application?",
"Exit Confirmation", JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE, null, null, null);
if (confirm == 0) {
System.exit(1);
}
}
};
frame.addWindowListener(exitListener);
frame.setDefaultCloseOperation(EXIT_ON_CLOSE);
frame.setJMenuBar(MenuBar);
frame.setPreferredSize(new Dimension(400, 300));
frame.setLocation(100, 100);
frame.pack();
frame.setVisible(true);
frame1.addWindowListener(exitListener);
frame1.setDefaultCloseOperation(EXIT_ON_CLOSE);
frame1.setPreferredSize(new Dimension(400, 300));
frame1.setLocation(500, 100);
frame1.pack();
frame1.setVisible(true);
}
private class ExitListener implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
int confirm = JOptionPane.showOptionDialog(frame,
"Are You Sure to Close this Application?",
"Exit Confirmation", JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE, null, null, null);
/*JOptionPane.showMessageDialog(null, "Whatever", "Whatever",
JOptionPane.ERROR_MESSAGE);
int confirm1 = JOptionPane.showOptionDialog(frame1,
"Are You Sure to Close this Application?",
"Exit Confirmation", JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE, null, null, null);*/
if (confirm == 0) {
frame.dispose();
}
}
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
ClosingFrame cf = new ClosingFrame();
}
});
}
}
编辑2
在当前API的实现中,两者之间没有任何区别
JFrame#dispose();
及
从当前JVM实例输入UsedMemory时,如果您想调用,这个/这些容器也会保持不变
try {
ImageIO.write(rendered, filetype, new File(file + "." + filetype));
frame.dispose();
logger.trace("Tried to dispose the frame");
} catch (IOException e) {
logger.fatal(e.getMessage());
} finally {
logger.debug("Try to dispose the frame");
frame.dispose();
}
然后代码将被执行,但是
1仅当在EDT上完成时,否则…EDT外部顶层容器将粘贴在屏幕上
或
2个顶级容器错过了方法finalize,然后将永远不会根据Windows NT/2000中的资源对其进行GC限制,这些资源处于活动状态…,然后控制台的输出是正确的,并保持重新绑定不变,直到存在当前JVM实例为止
3仅创建一个JFrame,然后通过重新使用此容器执行另一个用户操作
JFramegetContentPane.removeAll
JFrame.addwhich
Java7 JFramerevalidate;对于Java6,最好使用JPanel或ContentPane中的容器也可以这样做
JFramerepaint
编辑
为DefaultCloseOperations和JFramedisp设置的其余部分
你可以测试什么
您在这里发布的代码与此代码中来自JMenuItem的代码相同
import javax.swing.*;
import java.awt.*;
import java.awt.event.*;
import javax.swing.border.EmptyBorder;
public class ClosingFrame extends JFrame {
private JMenuBar MenuBar = new JMenuBar();
private JFrame frame = new JFrame();
private static final long serialVersionUID = 1L;
private JMenu File = new JMenu("File");
private JMenuItem Exit = new JMenuItem("Exit");
private JFrame frame1 = new JFrame();
public ClosingFrame() {
File.add(Exit);
MenuBar.add(File);
Exit.setBorder(new EmptyBorder(10, 10, 10, 10));
Exit.addActionListener(new ExitListener());
WindowListener exitListener = new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {
int confirm = JOptionPane.showOptionDialog(frame,
"Are You Sure to Close this Application?",
"Exit Confirmation", JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE, null, null, null);
if (confirm == 0) {
System.exit(1);
}
}
};
frame.addWindowListener(exitListener);
frame.setDefaultCloseOperation(EXIT_ON_CLOSE);
frame.setJMenuBar(MenuBar);
frame.setPreferredSize(new Dimension(400, 300));
frame.setLocation(100, 100);
frame.pack();
frame.setVisible(true);
frame1.addWindowListener(exitListener);
frame1.setDefaultCloseOperation(EXIT_ON_CLOSE);
frame1.setPreferredSize(new Dimension(400, 300));
frame1.setLocation(500, 100);
frame1.pack();
frame1.setVisible(true);
}
private class ExitListener implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
int confirm = JOptionPane.showOptionDialog(frame,
"Are You Sure to Close this Application?",
"Exit Confirmation", JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE, null, null, null);
/*JOptionPane.showMessageDialog(null, "Whatever", "Whatever",
JOptionPane.ERROR_MESSAGE);
int confirm1 = JOptionPane.showOptionDialog(frame1,
"Are You Sure to Close this Application?",
"Exit Confirmation", JOptionPane.YES_NO_OPTION,
JOptionPane.QUESTION_MESSAGE, null, null, null);*/
if (confirm == 0) {
frame.dispose();
}
}
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
ClosingFrame cf = new ClosingFrame();
}
});
}
}
编辑2
在当前API的实现中,两者之间没有任何区别
JFrame#dispose();
及
如果从当前JVM实例输入UsedMemory,此/这些容器将保持不变请尝试设置断点。还可以尝试调用frame.setVisiblefalse;在调用dispose之前。帧未被释放是什么意思?你如何测试这个?JFrame的默认关闭操作是什么?是JFrame.DISPOSE\u打开了吗?JFrame.EXIT_ON_CLOSE?你怎么知道它没有被处理?事实上,我看不出你已经设置了JFrame的默认关闭操作,所以它可能是JFrame.HIDE_ON_CLOSE的默认值,这可能不会做你期望它做的事情。为了更快地获得更好的帮助,发布一个。尝试设置一个断点。还可以尝试调用frame.setVisiblefalse;在调用dispose之前。帧未被释放是什么意思?你如何测试这个?JFrame的默认关闭操作是什么?是JFrame.DISPOSE\u打开了吗?JFrame.EXIT_ON_CLOSE?你怎么知道它没有被处理?事实上,我看不出你已经设置了JFrame的默认关闭操作,所以它可能是JFrame.HIDE_ON_CLOSE的默认值,这可能不是你期望它做的。为了更快地得到更好的帮助,发布一个。这有关系吗?我以为defaultCloseOperation只是用于关闭按钮。根据我几年前的经验,现在我想我记得我关了窗户,不是吗?是的,这很重要,因为你会发现,不,这不仅仅是关上按钮。它用于设置关闭此窗口时JVM的行为。Dispose只关闭窗口,不退出应用程序。再次,请检查此应用程序的API。但是,这会改变任何事情,我的应用程序仍然没有终止。@Roflcoptr:那么您已经将其设置为JFrame.EXIT\u on\u CLOSE,并且您的应用程序没有终止?完全正确。我尝试使用Window.getWindows,然后对所有窗口和Frame.getframes执行相同的处理。这有关系吗?我以为defaultCloseOperation只是用于关闭按钮。根据我几年前的经验,现在我想我记得我关了窗户,不是吗?是的,这很重要,因为你会发现,不,这不仅仅是关上按钮。它用于设置关闭此窗口时JVM的行为。Dispose只关闭窗口,不退出应用程序。再次,请检查此应用程序的API。但是,这会改变任何事情,我的应用程序仍然没有终止。@Roflcoptr:那么您已经将其设置为JFrame.EXIT\u on\u CLOSE,并且您的应用程序没有终止?完全正确。我尝试使用Window.getWindows,然后对所有窗口和Frame.getFrames执行处置