rabbitMQ springboot ssm 用户注册之短信发送实例(二)

4 篇文章 0 订阅
2 篇文章 0 订阅

rabbitMQ springboot ssm 用户注册之短信发送实例(二)

1 boot-user

1.1 目录结构

在这里插入图片描述

1.2 pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>boot_parent</artifactId>
        <groupId>com.liu</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>boot_user</artifactId>

    <dependencies>
        <dependency>
            <groupId>com.liu</groupId>
            <artifactId>boot_common</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>

        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-pool2</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>

                <executions>
                    <execution>
                        <goals>
                            <goal>repackage</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>

        </plugins>

    </build>
</project>
1.3 controller
 @PostMapping("/sendsms/{mobile}")
   public Result sendsms(@PathVariable ("mobile") String mobile){
       userService.sendSMS(mobile);

       return  new Result(true, CodeEnum.OK.getCode(),"验证码发送成功");

   }

   @PostMapping("/register/{code}")
    public Result registUser(@RequestBody User user,@PathVariable  String code){
       userService.addUser(user,code);

       return  new Result(true,CodeEnum.OK.getCode(),"用户注册成功");
   }

1.4 service
 public void sendSMS(String mobile){
            String random =RandomStringUtils.randomNumeric(6);
            log.info("验证码为:{}",random);

            redisTemplate.boundHashOps(RedisKeyEnum.CAPTCHA_CODE.getKey()).put(mobile,random);
     //保存验证码到redis
            Map map = new HashMap<>();
            map.put("code",random);
            map.put("mobile",mobile);
            rabbitTemplate.convertAndSend("CAPTCHA_CODE",map); //发送验证码到rabbitmq

        }


        public void addUser(User user, String code) {

            String syscode = (String) redisTemplate.boundHashOps(RedisKeyEnum.CAPTCHA_CODE.getKey()).get(user.getMobile());
            if (syscode ==null){
                throw new RuntimeException("请点击获取短信验证码");
            }
             if(!syscode.equals(code)){
                throw  new RuntimeException("您输入的验证码不正确,请确认后重新输入");
             }
             user.setId(idWorker.nextId()+"");
             userDao.save(user);
           

        }
1.5 DAO
/**
 * Created by Administrator on 2019/8/15 0015.
 *
 * concat() 字符串拼接很好用。记得这个是mysql的函数
 * 在param 导包的过程中,记得导包别倒错,要不然会查不到数据,或者报错
 * 这里还支持原生的sql。视情况而选择相应技术类型
 * 在使用原生的sql实例:
 *
 *
 */
public interface UserDao extends JpaRepository<User,String>,JpaSpecificationExecutor<User> {

// 这个是hibernate的方式
//  @Query(value = " select u  from  User u where u.loginName like concat('%',:query,'%') or u.cardId like concat('%',:query,'%') ")
//    这种方式代表使用原生的sql查询方式
//    @Query(value = "select u.*  from  boot_user u where u.login_name like concat('%',:query,'%') or u.card_id like concat('%',:query,'%')",nativeQuery = true)
//  占位符的使用
@Query(value = " select u  from  User u where u.loginName like concat('%',?1,'%') or u.cardId like concat('%',?1,'%') ")
    Page<User> findSearch(@Param("query") String query, Pageable pageable);




}
1.6 配置文件
server:
  port: 9006
  tomcat:
    uri-encoding: UTF-8
spring:
  application:
    name: boot-user
  redis:
    host: 192.168.85.198
    database: 0
    port: 6379
    timeout: 2000
    password:
    lettuce:
      pool:
        max-active: 20
        max-wait: -1
        max-idle: 8
        min-idle: 0
  datasource:
    driver-class-name: com.mysql.jdbc.Driver
    url: jdbc:mysql://192.168.85.198:3306/tensquare_user?
         characterEncoding=utf-8&&useSSL=false
    username: root
    password: 123456
  jpa:
    database: mysql
    show-sql: true
    generate-ddl: true
  http:
    encoding:
      charset: utf-8
      force: true
      enabled: true
  rabbitmq:
    host: 192.168.85.198
logging:
  level:
    org:
      hibernate:
        SQL: DEBUG

2 boot-sms

2.1 项目结构

在这里插入图片描述

2.2 pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>boot_parent</artifactId>
        <groupId>com.liu</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>boot_sms</artifactId>

    <dependencies>
        <dependency>
            <groupId>com.liu</groupId>
            <artifactId>boot_common</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>com.aliyun</groupId>
            <artifactId>aliyun-java-sdk-core</artifactId>
            <version>3.2.5</version>
        </dependency>
        <dependency>
            <groupId>com.aliyun</groupId>
            <artifactId>aliyun-java-sdk-dysmsapi</artifactId>
            <version>1.0.0</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>

        </dependency>

    </dependencies>


</project>

2.3 配置文件

这里的参数在阿里云上,你可以获取到。所以我就不提供了。小弟钱也不多。。。。

在这里插入图片描述

2.4 短信发送核心工具类
package com.liu.sms.util;
import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.dysmsapi.model.v20170525.QuerySendDetailsRequest;
import com.aliyuncs.dysmsapi.model.v20170525.QuerySendDetailsResponse;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsRequest;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.profile.DefaultProfile;
import com.aliyuncs.profile.IClientProfile;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.Environment;
import org.springframework.stereotype.Component;

import java.text.SimpleDateFormat;
import java.util.Date;
/**
 * 短信工具类
 * @author Administrator
 *
 */
@Component
public class SmsUtil {

    //产品名称:云通信短信API产品,开发者无需替换
    static final String product = "Dysmsapi";
    //产品域名,开发者无需替换
    static final String domain = "dysmsapi.aliyuncs.com";

    @Autowired
    private Environment env;

    // TODO 此处需要替换成开发者自己的AK(在阿里云访问控制台寻找)

    /**
     * 发送短信
     * @param mobile 手机号
     * @param template_code 模板号
     * @param sign_name 签名
     * @param param 参数
     * @return
     * @throws ClientException
     */
    public SendSmsResponse sendSms(String mobile,String template_code,String sign_name,String param) throws ClientException {
        String accessKeyId =env.getProperty("aliyun.sms.accessKeyId");
        String accessKeySecret = env.getProperty("aliyun.sms.accessKeySecret");
        //可自助调整超时时间
        System.setProperty("sun.net.client.defaultConnectTimeout", "10000");
        System.setProperty("sun.net.client.defaultReadTimeout", "10000");
        //初始化acsClient,暂不支持region化
        IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", accessKeyId, accessKeySecret);
        DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", product, domain);
        IAcsClient acsClient = new DefaultAcsClient(profile);
        //组装请求对象-具体描述见控制台-文档部分内容
        SendSmsRequest request = new SendSmsRequest();
        //必填:待发送手机号
        request.setPhoneNumbers(mobile);
        //必填:短信签名-可在短信控制台中找到
        request.setSignName(sign_name);
        //必填:短信模板-可在短信控制台中找到
        request.setTemplateCode(template_code);
        //可选:模板中的变量替换JSON串,如模板内容为"亲爱的${name},您的验证码为${code}"时,此处的值为
        request.setTemplateParam(param);
        //选填-上行短信扩展码(无特殊需求用户请忽略此字段)
        //request.setSmsUpExtendCode("90997");
        //可选:outId为提供给业务方扩展字段,最终在短信回执消息中将此值带回给调用者
        request.setOutId("yourOutId");
        //hint 此处可能会抛出异常,注意catch
        SendSmsResponse sendSmsResponse = acsClient.getAcsResponse(request);
        return sendSmsResponse;
    }

    public  QuerySendDetailsResponse querySendDetails(String mobile,String bizId) throws ClientException {
        String accessKeyId =env.getProperty("aliyun.sms.accessKeyId");
        String accessKeySecret = env.getProperty("aliyun.sms.accessKeySecret");
        //可自助调整超时时间
        System.setProperty("sun.net.client.defaultConnectTimeout", "10000");
        System.setProperty("sun.net.client.defaultReadTimeout", "10000");
        //初始化acsClient,暂不支持region化
        IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", accessKeyId, accessKeySecret);
        DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", product, domain);
        IAcsClient acsClient = new DefaultAcsClient(profile);
        //组装请求对象
        QuerySendDetailsRequest request = new QuerySendDetailsRequest();
        //必填-号码
        request.setPhoneNumber(mobile);
        //可选-流水号
        request.setBizId(bizId);
        //必填-发送日期 支持30天内记录查询,格式yyyyMMdd
        SimpleDateFormat ft = new SimpleDateFormat("yyyyMMdd");
        request.setSendDate(ft.format(new Date()));
        //必填-页大小
        request.setPageSize(10L);
        //必填-当前页码从1开始计数
        request.setCurrentPage(1L);
        //hint 此处可能会抛出异常,注意catch
        QuerySendDetailsResponse querySendDetailsResponse = acsClient.getAcsResponse(request);
        return querySendDetailsResponse;
    }
}
2.5 短信发送监听类
package com.liu.sms.linster;

import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;
import com.aliyuncs.exceptions.ClientException;
import com.liu.common.utils.JacksonUtil;
import com.liu.sms.util.SmsUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.amqp.rabbit.annotation.Queue;
import org.springframework.amqp.rabbit.annotation.RabbitHandler;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import java.util.HashMap;
import java.util.Map;

@Component
@RabbitListener(queues = "CAPTCHA_CODE")
@Slf4j
public class SmsListener {

    @Autowired
    private SmsUtil smsUtil;


    @Value("${aliyun.sms.template_code}")
    private String template_code;

    @Value("${aliyun.sms.sign_name}")
    private String sign_name;


//   消息处理
    @RabbitHandler
    public void sendSms(Map<String,String> map){

     log.debug("map:{}",map);
     log.debug("template_code:{}",template_code);
     log.debug("sign_name:{}",sign_name);
     Map sendMsg= new HashMap();
     sendMsg.put("code",map.get("code"));
//        try {
       //建议测试的时候,不要多次搞球,真的是要钱的
   SendSmsResponse sendSmsResponse= smsUtil.sendSms(map.get("mobile"),template_code,sign_name, JacksonUtil.toJSon(sendMsg));
//            //todo 这里需要考虑的两个地方,
//            // todo 一个是string param 这里的string是一个json对象的字符串 ,第二种情况是账户余额问题,当你的账户余额没有的时候,短信是发布出去的
//            // todo ,建议在这个位置对返回值进行一个判定,最常见isv.AMOUNT_NOT_ENOUGH 余额不足 、isv.MOBILE_NUMBER_ILLEGAL 非法手机号
//            // todo public SendSmsResponse sendSms(String mobile,String template_code,String sign_name,String param)
//
//        } catch (ClientException e) {
//            e.printStackTrace();
//
//        }
        log.info("短信发送成功:{}",map.get("code"));


    }

}

3 测试
3.1 用户发送验证码

在这里插入图片描述

3.2 用户注册

在这里插入图片描述

4 问题与建议

这里只是简单介绍下使用的方法,在生产中,我们还要注意以下问题:

消息的重复消费:

​ 思路:使用redis建一张表,把消费的消息id存进去,如果遇到重复消费的消息,我们可以丢弃。

消息顺序消费:

有的消息是必须要求消息顺序消费的,然而我们在多个消费者存在的情况下,怎么样才能保证消息按照顺序消费呢?

​ 思路:在消费过程中,我们可以将一组有顺序的消息,通过一个队列的方式,发送给消费者。而每个消费者只要关心,自己的队列消息,而保障消息顺序的一致性和数据的完整性。

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值