How does the 'actionPerformed' method get called without an explicit call to it?

无人久伴 提交于 2019-12-05 17:09:28

问题


I just started learning GUI with Swing and don't exactly understand how the actionPerformed method works. Consider the following code:

//code to create a button and change its text when clicked
public class simplegui implements ActionListener {
  JButton button;

  public static void main(String[] args) {
    simplegui gui=new simplegui();
    gui.go();
  }

  public void go() {
    JFrame frame=new Frame();
    button=new JButton("click Me");
    button.addActionListener(this);

    frame.getContentPane().add(button);
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.setSize(300,300);
    frame.setVisible(true);
  }

  public void actionPerformed(ActionEvent event) {
    button.setText("I've been clicked!");
  }
}

Shouldn't an object be created for a class before a method on it is evoked (except for static methods)?

When the button is clicked the actionPerformed method is called, but how? Where is the call made? I've implemented the interface ActionListener, but where is the code that knows that when an action occurs the 'ActionEvent' Object should be sent to the 'actionPerformed' method? Is it present in the Button class? Is the addActionListener method present in the Button class?

When I click the button, how is the system call action performed and where is the code that executes gui.actionPerformed()?

I followed Java concepts of OO, static etc. until now but this whole event driven programming is confusing.


回答1:


Each event is represented by an object that gives information about the event and identifies the event source. Event sources are often components or models, but other kinds of objects can also be event sources.

Here,the listener you registered, that is,

button.addActionListener(this);

gets added to a list of listeners, and when the JVM receives an event (click in this case), it calls the appropriate methods on all the listeners in the list.

How does this happen? Well, i think you should read about Callback mechanism in java.

You can also create your own listeners using Callback mechanism. Consider the below code:

The code is for a credit-card app simulation.In the below code, the pinChanged() method gets called automatically when the changePin() method is called.

public interface PinChangeListener {
    public void pinChanged();
}

public class CreditCard {
    public PinChangeListener pinChangeListener;

    private int pin;

    public changePin(int pin) {
        this.pin = pin;
        if (pinChangeListener != null) {
            pinChangeListener.pinChanged();
        }
    }
}

To connect a callback/listener to the credit card you just need to implement the PinChangeListener method:

creditCard.pinChangeListener = new PinChangeListener() {
    public void pinChanged() {
        System.out.println("The pin has been changed");
    }
};

Similarly, when you attach a listener to a button, the click is detected by JVM,(You probably don't want to go into how the click is detected!) and the actionPerformed() of that listener which is attached is called by the JVM for you. Hope this clears.




回答2:


But there is a specific call to this method, only it's not occurring in your code but rather in the JVM. A button push instigates internal events that leads the JVM to tell the button to notify all of its listeners that it has been pushed. This will cause the actionPerformed methods of all attached ActionListeners to be called.

To see information on how this works, first look at the Java API for the AbstractButton class where you'll find the method

protected void fireActionPerformed(ActionEvent event)

Where

Notifies all listeners that have registered interest for notification on this event type. The event instance is lazily created using the event parameter.

Then for further information, you will want to go beyond the Java API to the source code which can be found here. If you check out the Java 8.0 source code there, and look up javax then swing, then AbstractButton, you'll find a fireActionPerformed(ActionEvent event) method:

2002    protected void More ...fireActionPerformed(ActionEvent event) {
2003        // Guaranteed to return a non-null array
2004        Object[] listeners = listenerList.getListenerList();
2005        ActionEvent e = null;
2006        // Process the listeners last to first, notifying
2007        // those that are interested in this event
2008        for (int i = listeners.length-2; i>=0; i-=2) {
2009            if (listeners[i]==ActionListener.class) {
2010                // Lazily create the event:
2011                if (e == null) {
2012                      String actionCommand = event.getActionCommand();
2013                      if(actionCommand == null) {
2014                         actionCommand = getActionCommand();
2015                      }
2016                      e = new ActionEvent(AbstractButton.this,
2017                                          ActionEvent.ACTION_PERFORMED,
2018                                          actionCommand,
2019                                          event.getWhen(),
2020                                          event.getModifiers());
2021                }
2022                ((ActionListener)listeners[i+1]).actionPerformed(e);
2023            }
2024        }
2025    }


来源:https://stackoverflow.com/questions/25592125/how-does-the-actionperformed-method-get-called-without-an-explicit-call-to-it

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!