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来实现类似的功能。但为了加深对设计模式的理解,所以写了一个不是很纯正的观察者模式来模仿发布订阅的操作。

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

相关文章

  • mybatis-plus 自定义 Service Vo接口实现数据库实体与 vo 对象转换返回功能

    mybatis-plus 自定义 Service Vo接口实现数据库实体与 vo

    这篇文章主要介绍了mybatis-plus 自定义 Service Vo接口实现数据库实体与 vo 对象转换返回功能,本文通过实例图文相结合给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2024-08-08
  • Java转换流(InputStreamReader/OutputStreamWriter)的使用

    Java转换流(InputStreamReader/OutputStreamWriter)的使用

    本文主要介绍了Java转换流(InputStreamReader/OutputStreamWriter)的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-01-01
  • 在SpringBoot中使用@Value注解来设置默认值的方法

    在SpringBoot中使用@Value注解来设置默认值的方法

    Spring Boot提供了一种使用注解设置默认值的方式,即使用 @Value 注解,下面这篇文章主要给大家介绍了关于如何在SpringBoot中使用@Value注解来设置默认值的相关资料,需要的朋友可以参考下
    2023-10-10
  • 详解Spring Cloud Hystrix断路器实现容错和降级

    详解Spring Cloud Hystrix断路器实现容错和降级

    本篇文章主要介绍了详解Spring Cloud Hystrix断路器实现容错和降级,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • Java中的内存模型JMM详细解读

    Java中的内存模型JMM详细解读

    这篇文章主要介绍了Java中的内存模型JMM详细解读,Java 对内存的抽象模型如下,每个线程都有一块自己的私有内存(也称为工作内存),当线程使用变量时,会把主内存里面的变量复制到工作内存,线程读写变量时操作的是自己工作内存中的变量,需要的朋友可以参考下
    2023-12-12
  • java短网址服务(TinyURL)生成算法

    java短网址服务(TinyURL)生成算法

    这篇文章主要为大家详细介绍了java短网址服务生成算法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • springboot中jsp配置tiles全过程

    springboot中jsp配置tiles全过程

    这篇文章主要介绍了springboot中jsp配置tiles全过程,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-10-10
  • Java两种方式实现动态代理

    Java两种方式实现动态代理

    Java 在 java.lang.reflect 包中有自己的代理支持,该类(Proxy.java)用于动态生成代理类,只需传入目标接口、目标接口的类加载器以及 InvocationHandler 便可为目标接口生成代理类及代理对象。我们称这个Java技术为:动态代理
    2020-10-10
  • Spring Boot将@RestController误用于视图跳转问题解决

    Spring Boot将@RestController误用于视图跳转问题解决

    这篇文章主要为大家介绍了Spring Boot将@RestController误用于视图跳转问题解决方案详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-06-06
  • 最最常用的 100 个 Java类分享

    最最常用的 100 个 Java类分享

    这篇文章主要介绍了最最常用的 100 个 Java类分享,需要的朋友可以参考下
    2015-04-04

最新评论