Java实现观察者模式


前言

Spring 中实现一个观察者模式示例,调用 Controller 的接口可以触发观察者发送不通的邮件。

示例

1. controller

import com.idiudiu.api.service.observer.Subject;
import com.idiudiu.common.model.Result;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@Slf4j
@RequestMapping("/api/msg")
@RestController
@Api(value = "消息接口")
public class MessageController {
    @Autowired
    private Subject emailSubject;

    @ApiOperation(value = "发送测试邮件")
    @GetMapping("/sendTestEmail")
    public Result<String> sendTestEmail() {
        emailSubject.notifyObservers(null);
        return Result.responseSuccess(null);
    }
}

2. service

2.1 接口

/**
 * 主题接口
 */
public interface Subject {
    /**
     * 添加观察者
     */
    void attach(Observer observer);

    /**
     * 移除观察者
     */
    void detach(Observer observer);

    /**
     * 通知观察者, Object 可传任意参数
     */
    void notifyObservers(Object object);
}
/**
 * 观察者接口
 */
public interface Observer {
    /**
     * 行为
     */
    void action(Object object);
}

2.2 实现

这里 Observer 实现了 ApplicationRunner 接口,在服务启动时把自己注册到 Subject 中。

import com.google.common.collect.Lists;
import com.idiudiu.api.service.observer.Observer;
import com.idiudiu.api.service.observer.Subject;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
public class EmailSubject implements Subject {
    private static final List<Observer> OBSERVERS = Lists.newLinkedList();

    @Override
    public void attach(Observer observer) {
        OBSERVERS.add(observer);
    }

    @Override
    public void detach(Observer observer) {
        OBSERVERS.remove(observer);
    }

    @Override
    public void notifyObservers(Object object) {
        OBSERVERS.forEach(observer -> observer.action(object));
    }
}
import com.google.common.collect.Lists;
import com.idiudiu.api.service.observer.Observer;
import com.idiudiu.api.service.observer.Subject;
import com.idiudiu.common.utils.MailUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
public class DogObserver implements Observer, ApplicationRunner {
    @Autowired
    private Subject emailSubject;

    @Override
    public void action(Object object) {
        String subject = "邮件测试Dog";
        String content = "你好,我是Dog测试邮件!";
        List<String> toReceivers = Lists.newArrayList("xxx@qq.com");
        MailUtil.sendHtmlEmail(subject, content, toReceivers, null);
    }

    @Override
    public void run(ApplicationArguments args) throws Exception {
        emailSubject.attach(this);
    }
}
import com.google.common.collect.Lists;
import com.idiudiu.api.service.observer.Observer;
import com.idiudiu.api.service.observer.Subject;
import com.idiudiu.common.utils.MailUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
public class CatObserver implements Observer, ApplicationRunner {
    @Autowired
    private Subject emailSubject;

    @Override
    public void action(Object object) {
        String subject = "邮件测试Cat";
        String content = "你好,我是Cat测试邮件!";
        List<String> toReceivers = Lists.newArrayList("xxx@qq.com");
        MailUtil.sendHtmlEmail(subject, content, toReceivers, null);
    }

    @Override
    public void run(ApplicationArguments args) throws Exception {
        emailSubject.attach(this);
    }
}

附录

Java实现QQ邮件发送

  • 10
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
观察者模式是一种常见的设计模式,它的主要思想是在对象之间建立一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。 在Java观察者模式实现需要定义两个角色:Subject(主题)和Observer(观察者)。Subject负责维护一组观察者对象,以及通知它们状态的变化;而Observer则负责接收Subject的通知,并做出相应的处理。 下面是一个简单的Java实现: ```java import java.util.ArrayList; import java.util.List; // 主题接口 interface Subject { void attach(Observer observer); // 添加观察者 void detach(Observer observer); // 移除观察者 void notifyObservers(); // 通知所有观察者 } // 观察者接口 interface Observer { void update(); // 更新状态 } // 具体主题类 class ConcreteSubject implements Subject { private List<Observer> observers = new ArrayList<>(); public void attach(Observer observer) { observers.add(observer); } public void detach(Observer observer) { observers.remove(observer); } public void notifyObservers() { for (Observer observer : observers) { observer.update(); } } // 主题状态的变化 public void changeState() { // do something notifyObservers(); // 通知所有观察者 } } // 具体观察者类 class ConcreteObserver implements Observer { private String name; public ConcreteObserver(String name) { this.name = name; } public void update() { System.out.println(name + " has received the update."); } } // 测试 public class ObserverPatternDemo { public static void main(String[] args) { Subject subject = new ConcreteSubject(); Observer observer1 = new ConcreteObserver("Observer1"); Observer observer2 = new ConcreteObserver("Observer2"); Observer observer3 = new ConcreteObserver("Observer3"); subject.attach(observer1); subject.attach(observer2); subject.attach(observer3); ((ConcreteSubject) subject).changeState(); subject.detach(observer2); ((ConcreteSubject) subject).changeState(); } } ``` 在这个例子,我们定义了Subject和Observer两个接口,以及它们的具体实现类ConcreteSubject和ConcreteObserver。其ConcreteSubject负责维护一组观察者对象,并在状态变化时通知它们;而ConcreteObserver则负责接收Subject的通知,并做出相应的处理。 在测试代码,我们创建了一个ConcreteSubject对象,并向它添加了三个ConcreteObserver观察者。然后我们调用changeState()方法改变主题的状态,这会触发ConcreteSubject通知所有观察者。最后我们移除了一个观察者,并再次调用changeState()方法,这时只有两个观察者会收到通知。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值