java使用观察者模式异步短信/邮箱提醒用户群

 更新时间:2018年11月14日 08:38:25   作者:没有不忧伤的故事  
这篇文章主要为大家详细介绍了java使用观察者模式异步短信和邮箱提醒用户群,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

需求

用户中有人设置了账户余额达到阈值时,短信/邮箱进行提醒的服务。我们将需要在他账户余额阈值达到指定数值的时候进行短信/邮箱消息通知,允许账户余额阈值出现偏差的时候通知,如果某个用户48小时内已经短信/邮箱进行过通知了,那么将不再进行通知。

剖析

  • 存在两个主题:短信通知和邮箱通知
  • 存在两种观察者:设置了短信通知且账户余额到达阈值的用户,设置了邮箱通知且账户余额到达阈值的用户。
  • 用spring的定时器,每10分钟去数据库获取某个主题已经达到阈值且开始了该主题的提醒功能的用户
  • 用spring的@Asycn注解异步短信通知,邮箱通知的相关方法
  • 用redis设置用户短信/邮箱为键名,设置过期时间为48小时。如果获取不到该键值对,说明其在观察者行列

代码

观察者父类

/**
 * 订阅观察者
 * @author Administrator
 *
 */
@Component
//标志为多例
@Scope(value=ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public class SubscriberObserver implements Observer{

  private String email;
  private String phone;
  private String username;
  @Autowired
  UserFunctionService UserFunctionService;


  @Override
  public void update(Observable o, Object arg) {
    if(o instanceof EmailAlertSubject){
      UserFunctionService.alertUserEmail(email,username);
    }
    if(o instanceof PhoneAlertSubject){
      UserFunctionService.alertUserPhone(phone,username);
    }
  }

  public String getEmail() {
    return email;
  }

  public void setEmail(String email) {
    this.email = email;
  }

  public String getPhone() {
    return phone;
  }

  public void setPhone(String phone) {
    this.phone = phone;
  }

  public String getUsername() {
    return username;
  }

  public void setUsername(String username) {
    this.username = username;
  }

  public SubscriberObserver() {
    super();
    // TODO Auto-generated constructor stub
  }


}

主题

/**
 * email提醒主题
 * @author Administrator
 *
 */
@Component
public class EmailAlertSubject extends Observable{

  public void alert(){
     this.setChanged();
     //如果用拉的方式,这么调用
     this.notifyObservers();
  }
}
/**
 * 短信提醒主题
 * @author Administrator
 *
 */
@Component
public class PhoneAlertSubject extends Observable{

  public void alert(){
     this.setChanged();
     //如果用拉的方式,这么调用
     this.notifyObservers();
  }
}

定时器

/**
 * 定时给订阅了短信提醒和email提醒的用户服务
 * @author Administrator
 *
 */
@Component
public class TimeAlertTaskUtil {

  @Autowired
  CommonUserService commonUserService;
  @Autowired
  JedisConnectionFactory factory;
  @Autowired
  EmailAlertSubject emailSubject;
  @Autowired
  PhoneAlertSubject phoneSubject;


  private static final String emailKeyName = "emailAlert:";
  private static final String phoneKeyName = "phoneAlert:";

  /**
   * 定时获取需要email提醒的用户,每10分钟调用一次
   */
  @Scheduled(fixedDelay = 1000 * 60 * 10)
  public void alertEmailTask() {
    // 1.获取数据库中达到了阈值的用户
    List<User> emails = commonUserService.getUserAlertEmailAndName();
    // 2.查看redis中是否有达到阈值,且48小时已经通知的用户,将其排除在观察者行列,最终得出观察者队伍
    List<SubscriberObserver> informEmail = getInformObserver(emails);
    // 3.创建主题,添加观察者
    addObservers(emailSubject, informEmail);
    // 4.通知
    emailSubject.alert();

    // 5.将已经通知的观察者信息存储到reids内,设置过期时间为一天
    setRedisCache(emails);
    // 6.将观察者从主题中移除
    deleteObservers(emailSubject, informEmail);
  }

  /**
   * 定时获取需要短信提醒的用户,每10分钟调用一次
   * 
   */
  @Scheduled(fixedDelay = 1000 * 60 * 10)
  public void alertPhoneTask() {
    // 1.获取数据库中达到了阈值的用户
    List<User> phones = commonUserService.getUserAlertPhoneAndName();
    // 2.查看redis中是否有达到阈值,且今天已经通知的用户,将其排除在观察者行列,最终得出观察者队伍
    List<SubscriberObserver> informPhones = getInformObserver(phones);

    // 3.创建主题,添加观察者
    addObservers(phoneSubject, informPhones);
    // 4.通知
    phoneSubject.alert();
    // 5.将已经通知的观察者信息存储到reids内,设置过期时间为一天
    setRedisCache(phones);
    // 6.将观察者从主题中移除
    deleteObservers(phoneSubject, informPhones);
  }

  /**
   * ------------------------------------------------------------------------
   * -----------------------------------------------------
   **/
  /**
   * 过滤掉今天已经email提醒的用户,返回真正需要提醒的观察者列表
   * 
   * @param emails
   * @return
   */
  private List<SubscriberObserver> getInformObserver(
      List<User> users) {
    List<SubscriberObserver> obs = new ArrayList<SubscriberObserver>();
    Jedis jedis = factory.getConnection().getNativeConnection();
    for (User user : users) {
      String value;
      SubscriberObserver observer = (SubscriberObserver) SpringConfigTool
          .getBean("subscriberObserver");
      if (user.getEmail()!=null) {
        value = jedis.get(emailKeyName + user.getEmail());
        if (value == null || !value.equals("success")) {
          observer.setEmail(user.getEmail());
          observer.setUsername(user.getName());
          obs.add(observer);
        }
      } else {
        value = jedis.get(phoneKeyName + user.getPhone());
        if (value == null || !value.equals("success")) {
          observer.setPhone(user.getPhone());
          observer.setUsername(user.getName());
          obs.add(observer);
        }
      }
    }
    return obs;
  }

  /**
   * 将指定的观察者列表添加到指定的主题
   * 
   * @param subject
   * @param list
   */
  private void addObservers(Observable subject, List<SubscriberObserver> list) {
    for (SubscriberObserver obs : list) {
      subject.addObserver(obs);
    }
  }

  private void deleteObservers(Observable subject,
      List<SubscriberObserver> list) {
    for (SubscriberObserver obs : list) {
      subject.deleteObserver(obs);
    }
  }

  /**
   * 将列表的值作为键,存入redis,过期时间为48小时
   * 
   * @param list
   */
  private void setRedisCache(List<User> users) {
    Jedis jedis = factory.getConnection().getNativeConnection();
    for (User user : users) {
      if (user.getEmail()!=null) {
        jedis.set(emailKeyName + user.getEmail(), "success", "NX", "EX",
            60 * 60 * 24 * 2);
      } else {
        jedis.set(phoneKeyName + user.getPhone(), "success", "NX", "EX",
            60 * 60 * 24 * 2);
      }
    }
  }
}

总结

代码是不全面的,只是个示例而已。关于短信通知和邮箱通知的服务类和工具类并没有给出,因为里面涉及到一些隐私参数。所以关于异步通知示例代码没有,但使用Spring管理的@Async注解和在spring进行一定的配置即可,可以在我的另外一篇博客找到关于异步通知的示例代码。

事实上根据需求,可以使用redis的发布订阅,或者消息队列mq来实现类似的功能。但为了加深对设计模式的理解,所以写了一个不是很纯正的观察者模式来模仿发布订阅的操作。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Swing拆分窗格控件JSplitPane使用详解

    Swing拆分窗格控件JSplitPane使用详解

    这篇文章主要为大家详细介绍了Swing拆分窗格控件JSplitPane的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-11-11
  • Java案例之HashMap集合存储学生对象并遍历

    Java案例之HashMap集合存储学生对象并遍历

    这篇文章主要介绍了Java案例之HashMap集合存储学生对象并遍历,创建一个HashMap集合,键是学号(String),值是学生对象(Student),存储三个键值对元素并遍历,下文具体操作需要的朋友可以参考一下
    2022-04-04
  • Spring框架的ImportSelector详细解读

    Spring框架的ImportSelector详细解读

    这篇文章主要介绍了Spring框架的ImportSelector详细解读,Spring中一个非常重要的注解@Import中的ImportSelector接口的作用以及它到底有啥作用,也会捎带一部分源码说一下DeferredImportSelector是干啥的,需要的朋友可以参考下
    2024-01-01
  • elasticsearch聚合查询实践示例

    elasticsearch聚合查询实践示例

    这篇文章主要为大家介绍了elasticsearch聚合查询实践示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • Java文件快速copy复制实例代码

    Java文件快速copy复制实例代码

    这篇文章主要给大家介绍了关于Java文件快速copy复制的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-01-01
  • 使用Spring的JAVA Mail支持简化邮件发送功能

    使用Spring的JAVA Mail支持简化邮件发送功能

    这篇文章主要为大家详细介绍了使用Spring的JAVA Mail支持简化邮件发送功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-04-04
  • SpringBoot 整合Tess4J库实现图片文字识别案例详解

    SpringBoot 整合Tess4J库实现图片文字识别案例详解

    Tess4J是一个基于Tesseract OCR引擎的Java接口,可以用来识别图像中的文本,说白了,就是封装了它的API,让Java可以直接调用,今天给大家分享一个SpringBoot整合Tess4j库实现图片文字识别的小案例
    2023-10-10
  • SpringBoot整合SSO(single sign on)单点登录

    SpringBoot整合SSO(single sign on)单点登录

    这篇文章主要介绍了SpringBoot整合SSO(single sign on)单点登录,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • SpringMVC接收java.util.Date类型数据的2种方式小结

    SpringMVC接收java.util.Date类型数据的2种方式小结

    这篇文章主要介绍了使用SpringMVC接收java.util.Date类型数据的2种方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • 用Java实现OCR功能揭秘

    用Java实现OCR功能揭秘

    想知道如何用Java实现OCR功能吗?本指南将揭秘这一神秘技术,让你轻松掌握OCR的实现方法,无论是想提升技能还是解决问题,这篇指南都能帮助你一臂之力,需要的朋友可以参考下
    2023-12-12

最新评论