更改按钮颜色onclick而不使用最终修改器

时间:2015-08-31 16:24:03

标签: java swing user-interface jbutton local-variables

考虑以下代码,用于在单击按钮时更改按钮的颜色。

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);

    }
}

由于此错误,它无法正常工作:

  

不能引用封闭范围中定义的非最终局部变量

如果不将按钮标记为最终,我怎么能这样做?

2 个答案:

答案 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));