我有一个视口,并为其附加了一个更改侦听器。每当我在视口中滚动时,我的变更侦听器都会被调用大约四次。我不确定如何避免这种情况;我只希望通话发生一次?

最佳答案

无法解决此问题,JViewport将触发几个stateChanged事件,因为它提供有关许多属性更改的通知...

JavaDocs ...


  在每个通知的列表中添加一个ChangeListener
  视图的大小,位置或视口的范围大小的时间
  改变了。


在这一点上,由于我们不知道您要达到的目标,所以很难知道建议的内容,但是,如果必须使用ChangeListener,则可以建立合并机制。也就是说,您基本上是等到事件之间发生足够长的延迟后才响应,而不是响应每个事件。

例如...

public class DelayedChangeHandler implements ChangeListener {

    private Timer timer;
    private ChangeEvent last;

    public DelayedChangeHandler() {
        timer = new Timer(250, new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                stableStateChanged();
            }
        });
        timer.setRepeats(false);
    }

    @Override
    public void stateChanged(ChangeEvent e) {
        last = e;
        timer.restart();
    }

    protected void stableStateChanged() {
        System.out.println("Finally...");
    }

}


基本上,这是一个ChangeListener实现,它使用具有短延迟的非重复javax.swing.Timer。每次调用stateChanged时,计时器都会重新启动。最后,当计时器被允许“滴答”时,它调用stableStateChanged指示从引发上一个事件以来已经过去了足够的时间。

假设您不太在意事件的起因,而只是事件发生了...

一个可运行的例子...

import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.Timer;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

public class TestViewport {

    public static void main(String[] args) {
        new TestViewport();
    }

    public TestViewport() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                }

                JPanel pane = new JPanel() {
                    @Override
                    public Dimension getPreferredSize() {
                        return new Dimension(1000, 1000);
                    }
                };

                JScrollPane sp = new JScrollPane(pane);
                sp.getViewport().addChangeListener(new DelayedChangeHandler());
                sp.getViewport().addPropertyChangeListener(new PropertyChangeListener() {
                    @Override
                    public void propertyChange(PropertyChangeEvent evt) {
                        System.out.println(evt.getPropertyName());
                    }
                });

                JFrame frame = new JFrame("Testing");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.add(sp);
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class DelayedChangeHandler implements ChangeListener {

        private Timer timer;
        private ChangeEvent last;

        public DelayedChangeHandler() {
            timer = new Timer(250, new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    stableStateChanged();
                }
            });
            timer.setRepeats(false);
        }

        @Override
        public void stateChanged(ChangeEvent e) {
            last = e;
            timer.restart();
        }

        protected void stableStateChanged() {
            System.out.println("Finally...");
        }

    }

}

10-04 12:23