SpringBoot基于配置实现短信服务策略的动态切换

 更新时间:2025年04月25日 14:57:53   作者:颇有几分姿色  
这篇文章主要为大家详细介绍了SpringBoot在接入多个短信服务商(如阿里云、腾讯云、华为云)后,如何根据配置或环境切换使用不同的服务商,需要的小伙伴可以了解下

目标功能

多短信服务商(策略)接入

支持配置启用/禁用服务商

可配置默认短信服务商

支持动态切换(如按业务类型、环境等)

可扩展更多服务商/通道

示例配置(application.yml)

sms:
  default-provider: aliyun
  providers:
    aliyun:
      enabled: true
      app-id: xxx
      app-secret: yyy
    tencent:
      enabled: true
      app-id: aaa
      app-secret: bbb
    huawei:
      enabled: false
      app-id: 123
      app-secret: 456

配置类绑定

@Component
@ConfigurationProperties(prefix = "sms")
@Data
public class SmsProperties {

    private String defaultProvider;

    private Map<String, ProviderConfig> providers;

    @Data
    public static class ProviderConfig {
        private boolean enabled;
        private String appId;
        private String appSecret;
    }
}

短信发送策略接口

public interface SmsProvider {
    String getProviderKey(); // aliyun / tencent / huawei
    boolean isEnabled();     // 是否启用(可从配置读取)
    void sendSms(String phone, String message);
}

示例:阿里云 & 腾讯云 实现类

@Component
public class AliyunSmsProvider implements SmsProvider {

    private final SmsProperties smsProperties;

    public AliyunSmsProvider(SmsProperties smsProperties) {
        this.smsProperties = smsProperties;
    }

    @Override
    public String getProviderKey() {
        return "aliyun";
    }

    @Override
    public boolean isEnabled() {
        SmsProperties.ProviderConfig config = smsProperties.getProviders().get(getProviderKey());
        return config != null && config.isEnabled();
    }

    @Override
    public void sendSms(String phone, String message) {
        System.out.println("【阿里云】发送短信:" + phone + " - " + message);
        // 实际调用 SDK
    }
}

@Component
public class TencentSmsProvider implements SmsProvider {

    private final SmsProperties smsProperties;

    public TencentSmsProvider(SmsProperties smsProperties) {
        this.smsProperties = smsProperties;
    }

    @Override
    public String getProviderKey() {
        return "tencent";
    }

    @Override
    public boolean isEnabled() {
        SmsProperties.ProviderConfig config = smsProperties.getProviders().get(getProviderKey());
        return config != null && config.isEnabled();
    }

    @Override
    public void sendSms(String phone, String message) {
        System.out.println("【腾讯云】发送短信:" + phone + " - " + message);
        // 实际调用 SDK
    }
}

动态选择器(默认服务商 + 指定服务商)

@Component
public class SmsProviderRouter {

    private final SmsProperties smsProperties;
    private final Map<String, SmsProvider> providerMap;

    public SmsProviderRouter(List<SmsProvider> providers, SmsProperties smsProperties) {
        this.smsProperties = smsProperties;
        this.providerMap = providers.stream()
        .collect(Collectors.toMap(SmsProvider::getProviderKey, p -> p));
    }

    public void sendSms(String phone, String message, @Nullable String providerKey) {
        String key = (providerKey != null) ? providerKey : smsProperties.getDefaultProvider();
        SmsProvider provider = providerMap.get(key);

        if (provider == null || !provider.isEnabled()) {
            throw new IllegalStateException("短信服务商不可用: " + key);
        }

        provider.sendSms(phone, message);
    }
}

接口测试

@RestController
@RequestMapping("/sms")
public class SmsController {

    private final SmsProviderRouter router;

    public SmsController(SmsProviderRouter router) {
        this.router = router;
    }

    @PostMapping("/send")
    public String send(@RequestParam String phone,
                       @RequestParam String msg,
                       @RequestParam(required = false) String provider) {
        router.sendSms(phone, msg, provider);
        return "发送成功";
    }
}

测试效果

默认发送(配置的是 aliyun)

POST /sms/send?phone=13800138000&msg=测试一波

输出:

【阿里云】发送短信:13800138000 - 测试一波

指定腾讯云发送:

POST /sms/send?phone=13800138000&msg=测试一波&provider=tencent

输出:

【腾讯云】发送短信:13800138000 - 测试一波

总结

通过配置驱动 + 策略模式的组合,短信服务切换更灵活,具备:

  • 高扩展性(支持无限短信服务商)
  • 高可维护性(配置即策略)
  • 高可控性(启用、禁用、默认、切换快速搞定)

不仅适用于短信服务,还可拓展到:

  • 第三方支付(支付宝 / 微信 / Stripe)
  • 文件存储(阿里 OSS / 腾讯 COS)
  • 消息通知(钉钉 / 企业微信 / 飞书)

到此这篇关于SpringBoot基于配置实现短信服务策略的动态切换的文章就介绍到这了,更多相关SpringBoot动态切换短信服务内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java中拼接字符串String的N种方法总结

    Java中拼接字符串String的N种方法总结

    字符串拼接是我们在Java代码中比较经常要做的事情,就是把多个字符串拼接到一起,下面这篇文章主要给大家介绍了关于Java中拼接String的N种方法,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
    2022-02-02
  • Java实战之用Swing实现通讯录管理系统

    Java实战之用Swing实现通讯录管理系统

    今天给大家带来的是Java实战的相关知识,文章围绕着Swing实现通讯录管理系统展开,文中有非常详细的代码示例,需要的朋友可以参考下
    2021-06-06
  • java String.join()方法实例详解

    java String.join()方法实例详解

    String.join()是Java提供的一个实用方法,用于将多个字符串按照指定的分隔符连接成一个字符串,这一方法是Java8中引入的,极大地简化了字符串拼接的操作,本文给大家介绍java String.join()方法,感兴趣的朋友一起看看吧
    2025-06-06
  • springboot中的RestTemplate使用详解

    springboot中的RestTemplate使用详解

    这篇文章主要介绍了springboot中的RestTemplate使用详解,RestTemplate继承自InterceptingHttpAccessor并且实现了RestOperations接口,其中RestOperations接口定义了基本的RESTful操作,这些操作在RestTemplate中都得到了实现,需要的朋友可以参考下
    2023-09-09
  • SpringMVC处理数据输出的实例代码

    SpringMVC处理数据输出的实例代码

    这篇文章主要给大家介绍了关于SpringMVC处理数据输出的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-05-05
  • java实现电脑端扫描二维码

    java实现电脑端扫描二维码

    这篇文章主要为大家详细介绍了java实现电脑端扫描二维码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-10-10
  • 解决springboot 无法配置多个静态路径的问题

    解决springboot 无法配置多个静态路径的问题

    这篇文章主要介绍了解决springboot 无法配置多个静态路径的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • Java基础之反射

    Java基础之反射

    JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;反射是框架设计的灵魂,感兴趣的小伙伴可以参考阅读
    2023-03-03
  • springboot redis使用lettuce配置多数据源的实现

    springboot redis使用lettuce配置多数据源的实现

    这篇文章主要介绍了springboot redis使用lettuce配置多数据源的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • 详解Java并发包中线程池ThreadPoolExecutor

    详解Java并发包中线程池ThreadPoolExecutor

    ThreadPoolExecutor是Java语言对于线程池的实现。线程池技术使线程在使用完毕后不回收而是重复利用。如果线程能够复用,那么我们就可以使用固定数量的线程来解决并发问题,这样一来不仅节约了系统资源,而且也会减少线程上下文切换的开销
    2021-06-06

最新评论