
swing 是单线程 gui 框架,主线程(event dispatch thread, edt)负责处理事件和界面更新;若在 `actionperformed` 中使用阻塞式 `while (true)` 循环,将导致 edt 被永久占用,界面完全冻结、无响应。
在您的代码中,点击“New Frame”按钮后,程序进入一个无限、阻塞的 while (framerunning) 循环,且循环体内既未更新 framerunning,也未让出 EDT 控制权——这直接导致整个应用卡死。Swing 不支持“轮询式监听”,而应采用事件驱动 + 异步更新机制。
✅ 正确做法:使用 ChangeListener 监听 JSpinner 值变化,并在回调中安全更新目标窗口尺寸。
以下是重构后的关键改进(仅展示 Frame.java 的核心修改部分):
// 在 Frame 构造方法中,移除 while 循环,改用事件监听
private JFrame dynamicFrame; // 保存对外部窗口的引用
@Override
public void actionPerformed(ActionEvent e) {
if (e.getSource() == newframebutton) {
// 创建新窗口(仅一次)
dynamicFrame = new JFrame("Dynamic Frame");
dynamicFrame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
dynamicFrame.setBackground(Color.LIGHT_GRAY);
dynamicFrame.setLayout(new BorderLayout());
dynamicFrame.add(new JLabel("Resize me via spinners!", JLabel.CENTER), BorderLayout.CENTER);
// 初始尺寸
height = (int) heightinp.getValue();
width = (int) widthinp.getValue();
dynamicFrame.setSize(width, height);
dynamicFrame.setLocationRelativeTo(this);
dynamicFrame.setVisible(true);
// ✅ 关键:为两个 Spinner 添加 ChangeListener,实现「值变即更新」
modelheight.addChangeListener(e1 -> updateFrameSize());
modelwidth.addChangeListener(e1 -> updateFrameSize());
}
}
// 独立的尺寸同步方法,线程安全(始终在 EDT 中执行)
private void updateFrameSize() {
int newHeight = (int) heightinp.getValue();
int newWidth = (int) widthinp.getValue();
// 避免无意义重绘(尺寸未变时跳过)
if (dynamicFrame != null &&
(dynamicFrame.getHeight() != newHeight || dynamicFrame.getWidth() != newWidth)) {
dynamicFrame.setSize(newWidth, newHeight);
// 可选:调用 revalidate() / repaint() 确保布局刷新(对纯 setSize 通常非必需)
}
}⚠️ 注意事项:
- 绝不阻塞 EDT:任何耗时操作(如网络请求、文件读写、Thread.sleep() 或 while(true))都必须卸载到后台线程(如 SwingWorker),并用 SwingUtilities.invokeLater() 回到 EDT 更新 UI。
- 避免内存泄漏:若用户多次点击“New Frame”,应考虑复用窗口或显式关闭旧窗口(例如调用 dynamicFrame.dispose() 再新建)。
- Spinner 模型复用更佳:SpinnerNumberModel 本身已支持 ChangeListener,无需手动轮询 getValue()。
- setDefaultCloseOperation(3) 已过时:请改用 JFrame.DISPOSE_ON_CLOSE(值为 2)或 JFrame.EXIT_ON_CLOSE(值为 3,但仅适用于主窗口退出 JVM)。
总结:Swing 的响应式开发核心是「事件订阅 → 回调处理」,而非「主动轮询」。用 ChangeListener 替代 while 循环,既符合 Swing 线程模型,又能实现毫秒级尺寸响应,彻底解决界面冻结问题。











