SpringBoot实现多数据源的切换实践
976
2022-08-03
Java实现发送短信验证码+redis限制发送的次数功能(redis限制短信发送频率)
java实现短信验证码发送,由于我们使用第三方平台进行验证码的发送,所以首先,我们要在一个平台进行注册。这样的平台有很多,有的平台在新建账号的时候会附带赠几条免费短信。这里我仅做测试使用(具体哪个平台见参考三,很简单,注册账号就行,记得添加短信签名)。
另外,在实际项目中,如果有人恶意攻击,不停的发送短信验证码,就会造成很大的损失。故对发送次数做一定的限制就非常必要,这里我们限制一个手机号一天可以发多少短信和短信平台无关。
这里采用的是存redis来实现这一个功能。就是每次调用发送验证码这个接口都会判断手机号码是否在redis中存为key了。如果没有则创建一个key为手机号码value是1.因为redis中不支持数字所以将其变为了string类型。如果redis中已经有这个key了则将此key的值取出来加1再存进redis中。
代码实现:
pom.xml
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
RespBean.java
package com.lmc.bean;
public class RespBean {
private Integer status;
private String msg;
private Object obj;
public static RespBean build() {
return new RespBean();
}
public static RespBean ok(String msg) {
return new RespBean(200, msg, null);
}
public static RespBean ok(String msg, Object obj) {
return new RespBean(200, msg, obj);
}
public static RespBean error(String msg) {
return new RespBean(500, msg, null);
}
public static RespBean error(String msg, Object obj) {
return new RespBean(500, msg, obj);
}
private RespBean() {
}
private RespBean(Integer status, String msg, Object obj) {
this.status = status;
this.msg = msg;
this.obj = obj;
}
public Integer getStatus() {
return status;
}
public RespBean setStatus(Integer status) {
this.status = status;
return this;
}
public String getMsg() {
return msg;
}
public RespBean setMsg(String msg) {
this.msg = msg;
return this;
}
public Object getObj() {
return obj;
}
public RespBean setObj(Object obj) {
this.obj = obj;
return this;
}
}
RedisConfig.java
package com.lmc.config;
import com.fasterxml.jackson.annotation.jsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import java.net.UnknownHostException;
@Configuration
public class RedisConfig {
@Bean
@ConditionalOnMissingBean(name = "redisTemplate")
public RedisTemplate
RedisConnectionFactory redisConnectionFactory)
throws UnknownHostException {
Jackson2JsonRedisSerializer
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
RedisTemplate
template.setConnectionFactory(redisConnectionFactory);
template.setKeySerializer(jackson2JsonRedisSerializer);
template.setValueSerializer(jackson2JsonRedisSerializer);
template.setHashKeySerializer(jackson2JsonRedisSerializer);
template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.afterPropertiesSet();
return template;
}
@ConditionalOnMissingBean(StringRedisTemplate.class)
public StringRedisTehttp://mplate stringRedisTemplate(
StringRedisTemplate template = new StringRedisTemplate();
}
SMSController.java
package com.lmc.controller;
import com.lmc.bean.RespBean;
import com.lmc.utils.HttpClientUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.Calendar;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.TimeUnit;
/**
* @description:
* @Author: lmc
* @date: 2021/12/26 10:21
*/
@RestController
public class SMSController {
@Autowired
StringRedisTemplate stringRedisTemplate;
@RequestMapping("/send")
public RespBean sendSMS() {
String Uid = "xxxxxxxx";
String Key = "xxxxxxxxxxxxxxx";
String smsMob = "xxxxxxxxx";
String sendSMSCount = "sendSMSCount:" + smsMob;
if ("2".equals(stringRedisTemplate.opsForValue().get(sendSMSCount))) {
return RespBean.error("今天已达到发送短信验证码上限,请明天再试");
}
//短信内容
String smsText = "欢迎使用xx系统,验证码:8888";
Map maps = new HashMap();
maps.put("Uid", Uid);
maps.put("Key", Key);
maps.put("smsMob", smsMob);
maps.put("smsText", smsText);
String result = HttpClientUtils.sendHttpPost("http://utf8.sms.webchinese.cn", maps);
int i = Integer.parseInt(result);
if (i > 0) {
if (stringRedisTemplate.opsForValue().get(sendSMSCount) == null) {
stringRedisTemplate.opsForValue().set(sendSMSCount, "1", getEndTime(), TimeUnit.MILLISECONDS);
} else {
String value = stringRedisTemplate.opsForValue().get(sendSMSCount);
int times = Integer.parseInt(value) + 1;
String timesStr = String.valueOf(times);
stringRedisTemplate.opsForValue().set(sendSMSCount, timesStr, getEndTime(), TimeUnit.MILLISECONDS);
}
return RespBean.ok("发送成功");
return RespBean.ok("发送失败");
}
//获取当前时间到今天结束时间所剩余的毫秒数:
public static long getEndTime() {
//获取当前时间的毫秒数
long time = new java.util.Date().getTime();
//获取到今天结束的毫秒数
Calendar todayEnd = Calendar.getInstance();
todayEnd.set(Calendar.HOUR_OF_DAY, 23); // Calendar.HOUR 12小时制。HOUR_OF_DAY 24小时制
todayEnd.set(Calendar.MINUTE, 59);
todayEnd.set(Calendar.SECOND, 59);
todayEnd.set(Calendar.MILLISECOND, 999);
long endTime = todayEnd.getTimeInMillis();
//这里endTime-time获取的是到23:59:59:999的毫秒数。再加1才是到24点整的毫秒数
return endTime-time+1;
}
HttpClientUtils.java(HttpClient工具类,可以复用)
package com.lmc.utils;
import java.io.IOException;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.apache.http.NameValuePair;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.methods.HttpRequestBase;
import org.apache.http.conn.ssl.DefaultHostnameVerifier;
import org.apache.http.conn.util.PublicSuffixMatcher;
import org.apache.http.conn.util.PublicSuffixMatcherLoader;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class HttpClientUtils {
private static Logger logger = LoggerFactory.getLogger(HttpClientUtils.class);
// 链接相关参数
private static int socketTimeout = 15000;
private static int connectTimeout = 15000;
private static int connectionRequestTimeout = 15000;
private static RequestConfig requestConfig = null;
// 连接池相关参数
private static int connMgrMaxTotal = 100;
private static int connMgrMaxPerRoute = 50;
private static PoolingHttpClientConnectionManager connMgr = null;
static {
requestConfig = RequestConfig.custom().setSocketTimeout(socketTimeout).setConnectTimeout(connectTimeout).setConnectionRequestTimeout(connectionRequestTimeout).build();
connMgr = new PoolingHttpClientConnectionManager();
connMgr.setDefaultMaxPerRoute(connMgrMaxPerRoute);
connMgr.setMaxTotal(connMgrMaxTotal);
}
private static String doHttp(HttpRequestBase httpRequestBase) {
CloseableHttpClient httpClient = null;
CloseableHttpResponse response = null;
String responseContent = null;
try {
// 创建默认的httpClient实例.
String scheme = httpRequestBase.getURI().getScheme();
if (scheme.equalsIgnoreCase("https")) {
PublicSuffixMatcher publicSuffixMatcher = PublicSuffixMatcherLoader.load(new URL(httpRequestBase.getURI().toString()));
DefaultHostnameVerifier hostnameVerifier = new DefaultHostnameVerifier(publicSuffixMatcher);
httpClient = HttpClients.custom().setSSLHostnameVerifier(hostnameVerifier).setConnectionManager(connMgr).build();
//httpClient = HttpClients.custom().setSSLHostnameVerifier(hostnameVerifier).build();
} else if (scheme.equalsIgnoreCase("http")) {
httpClient = HttpClients.custom().setConnectionManager(connMgr).build();
//httpClient = HttpClients.createDefault();
} else {
throw new IllegalArgumentException("url的scheme错误,必须是http或者https! ");
}
httpRequestBase.setConfig(requestConfig);
// 执行请求
response = httpClient.execute(httpRequestBase);
// 如果这里有必要获取的是其他资料都可以在这里进行逻辑处理
responseContent = EntityUtils.toString(response.getEntity(), "UTF-8");
return responseContent;
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
// 关闭连接,释放资源
if (response != null) {
// EntityUtils.consume(response.getEntity());
response.close();
}
// 这里不能关闭httpClient,这个会关链接池
//if (httpClient != null) {
// httpClient.close();
//}
} catch (IOException e) {
e.printStackTrace();
}
return responseContent;
/**
* sendHttpGet(url)
* @param url
* @return
*/
public static String sendHttpGet(String url) {
return doHttp(new HttpGet(url));
* sendHttpGet()
* @param param key1=value1&key2=value2&key3=value3
public static String sendHttpGet(String url, String param) {
// 创建httpGet
HttpGet httpGet = new HttpGet(url + '?' + param);
return doHttp(httpGet);
* sendHttpPost()
public static String sendHttpPost(String url, String param) {
// 创建httpPost
HttpPost httpPost = new HttpPost(url);
StringEntity stringEntity = new StringEntity(param, "UTF-8");
stringEntity.setContentType("application/x-www-form-urlencoded");
http://httpPost.setEntity(stringEntity);
return doHttp(httpPost);
* sendHttpGet
* @param param 是个map
public static String sendHttpGet(String url, Map
String paramStr = "";
for (String key : param.keySet()) {
String tmp = "";
tmp = "&" + key + "=" + param.get(key);
paramStr += tmp;
paramStr = paramStr.substring(1);
HttpGet httpGet = new HttpGet(url + '?' + paramStr);
return doHttp(httpGet);
* sendHttpPost
* @param param 是个map
public static String sendHttpPost(String url, Map
// 创建参数队列
List
for (String key : param.keySet()) {
nameValuePairs.add(new BasicNameValuePair(key, param.get(key)));
}
try {
httpPost.setEntity(new UrlEncodedFormEntity(nameValuePairs, "UTF-8"));
} catch (Exception e) {
e.printStackTrace();
return doHttp(httpPost);
public static String sendHttpPostJson(String url, String json) {
// StringEntity stringEntity = new StringEntity(param, "UTF-8");
// stringEntity.setContentType("application/json");
// stringEntity.setContentEncoding("UTF-8");
StringEntity stringEntity = new StringEntity(json, ContentType.create("application/json", "UTF-8"));
public static void main(String[] args) {
String url = "http://api.crpay.com/payapi/gateway";
String param = JcbROQQ"merchant_no=TOF00001&method=unified.trade.pay&version=1.0";
Map map = new HashMap
map.put("merchant_no", "TOF00001");
map.put("method", "unified.trade.pay");
map.put("version", "1.0");
// 这个工具是走的链接池,但是在关闭httpClient会关闭连接池的地方已经注销
//System.out.println(HttpClientUtils.sendHttpPost(url, map));
//System.out.println(HttpClientUtils.sendHttpPost(url, param));
//System.out.println(HttpClientUtils.sendHttpGet(url, map));
System.out.println(HttpClientUtils.sendHttpGet("https://baidu.com"));
System.out.println(HttpClientUtils.sendHttpGet("http://baidu.com/s?wd=aaa"));
Map map2 = new HashMap
map2.put("wd", "aaa");
System.out.println(HttpClientUtils.sendHttpGet("http://baidu.com/s",map2));
// doHttp是静态私有方法,不能使用多次,会报Connection pool shut down
System.out.println(HttpClientUtils.doHttp(new HttpGet("http://baidu.com/s?wd=aaa")));
System.out.println(HttpClientUtils.doHttp(new HttpGet("https://baidu.com/")));
System.out.println(HttpClientUtils.sendHttpGet("https://cnblogs.com/hugo-zhangzhen/p/6858013.html"));
System.out.println(HttpClientUtils.sendHttpGet("https://cnblogs.com/hugo-zhangzhen/p/6739658.html"));
System.out.println(HttpClientUtils.sendHttpGet("https://cnblogs.com/hugo-zhangzhen/p/6737810.html"));
System.out.println(HttpClientUtils.sendHttpGet("http://blog.csdn.net/xiechengfa/article/details/42016153"));
}
application.properties
# 配置redis
spring.redis.host=127.0.0.1
spring.redis.port=6379
spring.redis.password=123456
项目结果如下:
结果展示:
使用postman调用接口,超过2次后,显示如下。
在具体项目中的流程一般如下:
①构造手机验证码,需要生成一个6位的随机数字串;
②找短信平台获取使用接口向短信平台发送手机号和验证码,然后短信平台再把验证码发送到制定手机号上;
③将手机号验证码、操作时间存入Session中,作为后面验证使用;
④接收用户填写的验证码、手机号及其他注册数据;
⑤对比提交的验证码与Session中的验证码是否一致,同时判断提交动作是否在有效期内;
⑥验证码正确且在有效期内,请求通过,处理相应的业务。
参考:
接收短信验证码条数限制(java发送短信验证码限制) - 简书
Java如何实现短信验证码功能? - 知乎
Java 实现手机发送短信验证码 - 胖头陀春天 - 博客园
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~