Java监听器与观察者模式

Java中的监听器(Listener)和观察者模式(Observer Pattern)都是用于处理对象间的事件通知和响应的设计模式。它们的目的是在对象之间建立一种松散的耦合,使得一个对象的状态变化可以通知到其他对象,并且其他对象能够相应地作出变化。
首先我们先用两个实例来感受一下:

观察者模式:

观察者模式是一种行为设计模式,它定义了一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖于它的对象都得到通知并被自动更新。

javafx combobox 设置监听_监听器

在Java中,观察者模式通常使用java.util.Observable类和java.util.Observer接口来实现。被观察的对象继承Observable类,而观察者实现Observer接口。当被观察对象的状态发生改变时,它会调用notifyObservers()方法,通知所有注册的观察者。

import java.util.Observable;
import java.util.Observer;
// 被观察的对象
class MyObservable extends Observable {
    private int data;

    public int getData() {
        return data;
    }

    public void setData(int data) {
        this.data = data;
        setChanged(); // 表示状态已改变
        notifyObservers(data); // 通知观察者
    }
}

// 观察者
class MyObserver implements Observer {
    @Override
    public void update(Observable o, Object arg) {
        if (o instanceof MyObservable) {
            System.out.println("Data changed to: " + arg);
        }
    }
}

public class ObserverPatternExample {
    public static void main(String[] args) {
        MyObservable observable = new MyObservable();
        MyObserver observer = new MyObserver();

        observable.addObserver(observer);

        observable.setData(42); // 触发通知
    }
}

这个代码比较简单,在被观察对象 MyObservable发生改变时,会通知观察者,监听此对象的观察者 MyObserver会同步做出处理。
其中有一步observable.addObserver(observer); 这个是将被观察对象让观察者检测到
其内部实现主要在Observable
源码如下所示

public class Observable {
    private boolean changed = false;
    private Vector<Observer> obs = new Vector();  //观察者列表

    public Observable() {
    }

    public synchronized void addObserver(Observer o) {  //添加观察者
        if (o == null) {
            throw new NullPointerException();
        } else {
            if (!this.obs.contains(o)) {
                this.obs.addElement(o);
            }

        }
    }
    public synchronized void deleteObserver(Observer o) {  //删除观察者
        this.obs.removeElement(o);
    }
    public void notifyObservers() {   //通知观察者
        this.notifyObservers((Object)null);
    }
    public void notifyObservers(Object arg) { //通知观察者带参执行
        Object[] arrLocal;
        synchronized(this) {
            if (!this.changed) {
                return;
            }
            arrLocal = this.obs.toArray();
            this.clearChanged();
        }
        for(int i = arrLocal.length - 1; i >= 0; --i) {
            ((Observer)arrLocal[i]).update(this, arg);
        }

    }

    public synchronized void deleteObservers() { //删除所有观察者
        this.obs.removeAllElements();
    }

    protected synchronized void setChanged() {/当改变是可以通知观察者执行
        this.changed = true;
    }

    protected synchronized void clearChanged() {//当改变是可以通知观察者不执行
        this.changed = false;
    }

    public synchronized boolean hasChanged() { //获取观察者是否可以执行
        return this.changed;
    }

    public synchronized int countObservers() {//返回当前观察者个数
        return this.obs.size();
    }
}

通过源码可以看出addObserver方法可以将观察者加入到这个被观察者的属性中,通过维护一个Vector来维护所有的观察者,观察者实现Observer接口的update方法,来执行通知的方法。

监听器模式:

监听器模式并不是一个新的设计模式,它是观察者模式在特定场景下的一种改造和应用。通常,观察者模式的主题在通知观察者时,通知中不包含任何信息。如果这个过程中携带了一些其他信息,那么主题本身就成为了事件源,而携带信息的封装类就成为了事件。此时的观察者模式,也就升级为监听器了。监听器模式是观察者模式的另一种形态。
监听器模式通常包含三个角色:事件源、事件对象、事件监听器。

在Java中,监听器模式通常通过接口和事件对象来实现。
具体实现如下:

// 监听器接口
interface MyListener {
    void onDataChanged(int newData);
}
// 发布者类
class EventSource {
    private MyListener listener;

    public void setListener(MyListener listener) {
        this.listener = listener;
    }

    public void fireEvent(int newData) {
        if (listener != null) {
            listener.onDataChanged(newData);
        }
    }
}

// 订阅者类
class MySubscriber implements MyListener {
    @Override
    public void onDataChanged(int newData) {
        System.out.println("Data changed to: " + newData);
    }
}

public class ListenerPatternExample {
    public static void main(String[] args) {
        EventSource eventSource = new EventSource();
        MySubscriber subscriber = new MySubscriber();
        eventSource.setListener(subscriber);
        eventSource.fireEvent(42); // 触发事件
    }
}