考虑以下代码,用于在单击按钮时更改按钮的颜色。
import java.awt.Color;
import java.awt.event.ActionEvent;
import javax.swing.AbstractAction;
import javax.swing.JButton;
import javax.swing.JFrame;
public class ChangeColor {
public static void main(String[] args) {
JButton button = new JButton();
button.setBackground(Color.BLUE);
button.addActionListener(new AbstractAction() {
private static final long serialVersionUID = 1L;
@Override
public void actionPerformed(ActionEvent e) {
button.setBackground(Color.RED);
}
});
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(button);
frame.setSize(200, 200);
frame.setVisible(true);
}
}
由于此错误,它无法正常工作:
不能引用封闭范围中定义的非最终局部变量
如果不将按钮标记为最终,我怎么能这样做?
答案 0 :(得分:3)
一种解决方案是从ActionEvent参数中获取JButton的引用:
@Override
public void actionPerformed(ActionEvent e) {
// the source here refers to the object that provoked this method to occur
JButton sourceBtn = (JButton) e.getSource();
sourceBtn.setBackground(Color.RED);
}
此外,您可以将按钮设为字段,而不是局部变量 你也可以宣布它为最终版。
请注意,您永远不会使用诸如您发布的内容之类的代码,因为大多数非平凡的GUI程序都不会像您一样在静态主方法中创建,而是会在符合OOP的GUI类中完成。
例如:
import java.awt.Color;
import java.awt.event.ActionEvent;
import javax.swing.*;
public class ButtonColorChange extends JPanel {
private JButton button = new JButton(new ButtonAction("Press Me"));
public ButtonColorChange() {
add(button);
}
private static void createAndShowGui() {
JFrame frame = new JFrame("ButtonColorChange");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(new ButtonColorChange());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
createAndShowGui();
}
});
}
}
class ButtonAction extends AbstractAction {
private static final Color COLOR_1 = Color.BLUE;
private static final Color COLOR_2 = Color.RED;
public ButtonAction(String name) {
super(name);
int mnemonic = (int) name.charAt(0);
putValue(MNEMONIC_KEY, mnemonic);
}
@Override
public void actionPerformed(ActionEvent e) {
AbstractButton btn = (AbstractButton) e.getSource();
Color c = btn.getBackground();
c = c == COLOR_1 ? COLOR_2 : COLOR_1;
btn.setBackground(c);
}
}
编辑有关您的更多问题:
实际上我的问题中的代码只是一个例子,真正的代码就是你的代码。
然后更好地展示真实代码,或者最好是模拟真实代码的最小可编译runnable示例。
(a)最好使用e.getSource()或将按钮设为字段?
或者。如果要在多个按钮上使用相同的Action,并且您只对按下的按钮感兴趣,请使用getSource()
。如果您的Action或ActionListener是一个内部类,那么一个字段可以正常工作。如果没有,那么您可以根据问题的另一个答案通过构造函数参数将变量传递给Action。
(b)您说也可以将其声明为final,但我收到此错误:本地变量可能尚未初始化。我错了什么?
如果没有您的真实代码,很难说,但很可能您没有在宣布该按钮的同时初始化该按钮。
(c)我的GUI程序我没有从一个线程启动框架(id est我在main()中编写createAndShowGui()的代码,没有线程)。有什么区别?
我的代码保证我的GUI将在Swing事件线程上启动,这应该是为了线程安全而做的。
答案 1 :(得分:0)
您可以将Button作为构造函数参数提供给侦听器。
class ButtonColorAction extends AbstractAction {
private JButton button;
public ButtonColorAction(JButton button) {
this.button = button;
}
@Override
public void actionPerformed(ActionEvent e) {
button.setBackground(Color.RED);
}
}
并使用它:
button.addActionListener(new ButtonColorAction(button));