java-设计模式-创建模式-观察者模式observer

本文介绍观察者模式的应用场景及其实现方式,通过三个实例详细解释如何实现对象间的依赖关系,当一个对象状态改变时,所有依赖对象都能及时更新。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

java-设计模式-创建模式-观察者模式observer


一、什么时候用
    当某个对象的状态发生改变时,你仍然需要对象之间能互相通信。但是出于各种原因,你也许并不愿意因为代码环境的改变而对代码做大的修改。当一个对象的状态发生改变时,你如何通知其他对象?是否需要一个动态方案――一个就像允许脚本的执行一样,允许自由连接的方案。
    观测模式:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时, 所有依赖于它的对象都得到通知并被自动更新。

二、实例
实例一:

Subject接口

<span style="font-size:18px;">public interface Subject {
    public void registerObserver(Observer o);
    public void removeObserver(Observer o);
    public void notifyAllObservers();
}</span>


Observer接口

<span style="font-size:18px;">public interface Observer {
    public void update(Subject s);
}</span>

Hunter类实现了Subject接口

<span style="font-size:18px;">import java.util.ArrayList;
 
public class HeadHunter implements Subject{
 
    //define a list of users, such as Mike, Bill, etc.
    private ArrayList<Observer> userList;
    private ArrayList<String> jobs;
 
    public HeadHunter(){
        userList = new ArrayList<Observer>();
        jobs = new ArrayList<String>();
    }
 
    @Override
    public void registerObserver(Observer o) {
        userList.add(o);
    }
 
    @Override
    public void removeObserver(Observer o) {}
 
    @Override
    public void notifyAllObservers() {
        for(Observer o: userList){
            o.update(this);
        }
    }
 
    public void addJob(String job) {
        this.jobs.add(job);
        notifyAllObservers();
    }
 
    public ArrayList<String> getJobs() {
        return jobs;
    }
 
    public String toString(){
        return jobs.toString();
    }
}</span>


JobSeeker是一个观察者:

<span style="font-size:18px;">public class JobSeeker implements Observer {
 
    private String name;
 
    public JobSeeker(String name){
        this.name = name;
    }
    @Override
    public void update(Subject s) {
        System.out.println(this.name + " got notified!");
        //print job list
        System.out.println(s);
    }
}</span>

开始使用:

<span style="font-size:18px;">public class Main {
    public static void main(String[] args) {
        HeadHunter hh = new HeadHunter();
        hh.registerObserver(new JobSeeker("Mike"));
        hh.registerObserver(new JobSeeker("Chris"));
        hh.registerObserver(new JobSeeker("Jeff"));
 
        //每次添加一个个job,所有找工作人都可以得到通知。
        hh.addJob("Google Job");
        hh.addJob("Yahoo Job");
    }
}</span>



实例二:

 首先定义抽象的观察者:

//抽象观察者角色
<span style="font-size:18px;">public interface Watcher{
    public void update(String str);

}</span>

然后定义抽象的主题角色,即抽象的被观察者,在其中声明方法(添加、移除观察者,通知观察者):

//抽象主题角色,watched:被观察
<span style="font-size:18px;">public interface Watched{
    public void addWatcher(Watcher watcher);

    public void removeWatcher(Watcher watcher);

    public void notifyWatchers(String str);

}</span>


然后定义具体的观察者:
<span style="font-size:18px;">public class ConcreteWatcher implements Watcher{

    @Override
    public void update(String str)
    {
        System.out.println(str);
    }

}</span>

之后是具体的主题角色: 
<span style="font-size:18px;">import java.util.ArrayList;
import java.util.List;

public class ConcreteWatched implements Watched
{
    // 存放观察者private List<Watcher> list = new ArrayList<Watcher>();

    @Override
    public void addWatcher(Watcher watcher)
    {
        list.add(watcher);
    }

    @Override
    public void removeWatcher(Watcher watcher)
    {
        list.remove(watcher);
    }

    @Override
    public void notifyWatchers(String str)
    {
        // 自动调用实际上是主题进行调用的for (Watcher watcher : list)
        {
            watcher.update(str);
        }
    }

}</span>

编写测试类:
<span style="font-size:18px;">public class Test
{
    public static void main(String[] args)
    {
        Watched girl = new ConcreteWatched();
        
        Watcher watcher1 = new ConcreteWatcher();
        Watcher watcher2 = new ConcreteWatcher();
        Watcher watcher3 = new ConcreteWatcher();
        
        girl.addWatcher(watcher1);
        girl.addWatcher(watcher2);
        girl.addWatcher(watcher3);
        
        girl.notifyWatchers("开心");
    }

}</span>

实例三: JDK实例
Client
    测试类。
NewsPublisher
    相当于ConcreteSubject角色。该类继承相当于Subject角色的java.util.Observable类。
SubscriberObserver
    相当于ConcreteObserver角色。该类实现了相当于Observer角色的java.util.Observer接口。当NewsPublisher对象发生变化时得到通知,并向订阅者发送订阅邮件。
ManagerObserver
    相当于ConcreteObserver角色。该类实现了相当于Observer角色的java.util.Observer接口。当NewsPublisher对象发生变化时得到通知,并向管理者发送消息邮件。
News
    封装了新闻数据的类,与Observer模式无直接关系。

代码:
<span style="font-size:18px;">import java.util.Observable;
import java.util.Observer;

public class Client {

    /**
     * Test Observer Pattern
     * --
     * 
     */
    public static void main(String[] args) {
        NewsPublisher publisher = new NewsPublisher();
        //添加观察者对象
        publisher.addObserver(new SubscriberObserver());
        publisher.addObserver(new ManagerObserver());
        
        //发布新闻,触发通知事件
        publisher.publishNews("Hello news", "news body");
    }
}


/**
* Subject ROLE
* NewsPublisher: news publisher
*
*/
class NewsPublisher extends Observable {
    public void publishNews(String newsTitle, String newsBody) {
        News news = new News(newsTitle, newsBody);
        setChanged();       //通过setChanged()方法标明对象的状态已发生变化
        
        System.out.println("News published:" + newsTitle);
        this.notifyObservers(news);   //通知各Observer,并发送一个名为news对象的消息
        
        //other process ... such as save news to database
    }
}

/**
* Observer ROLE
* 
*/
class SubscriberObserver implements Observer {
    
    public void update(Observable observee, Object param) {
        if (param instanceof News) {
            mail2Subscriber((News)param);
        }
    }
    
    private void mail2Subscriber(News news) {
        System.out.println("Mail to subscriber. A news published with title:" + news.getTitle());
    }
}

/**
* Observer ROLE
* 
*/
class ManagerObserver implements Observer {
    
    public void update(Observable observee, Object param) {
        if (param instanceof News) {
            mail2Manager((News)param);
        }
    }
    
    private void mail2Manager(News news) {
        System.out.println("Mail to Manager. A news published with title:" + news.getTitle());
    }
}

//data entity
class News {
    private String title;
    private String body;
    
    public News(String title, String body) {
        this.title = title;
        this.body = body;
    }
    
    public String getBody() {
        return body;
    }
    public void setBody(String body) {
        this.body = body;
    }
    public String getTitle() {
        return title;
    }
    public void setTitle(String title) {
        this.title = title;
    }
}</span>

执行Client,输出结果:
C:\Observer>javac *.java
C:\Observer>java Client
News published:Hello news
Mail to Manager. A news published with title:Hello news
Mail to subscriber. A news published with title:Hello news






实例二:

 首先定义抽象的观察者:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值