浅谈对于登录注册业务的思路以及部分代码的实现

近期开始了与移动端项目的开发,由于我个人是后台方面的所以肯定避免不了要实现每个app都应该具有的功能:登录与注册,我一开始写也是感觉非常的没有头绪,所以这里写一篇博客分享给大家如何实现一个基本的登录与注册的功能,希望有意见也积极提出!
那么首先先说一下标题提到的几个词汇的功能:

token

如果我们每次关闭客户端再次打开的时候都需要进行登录操作甚至是每次访问controller之前都需要进行是否登录的判断,那么每次如果都要访问持久化数据库的话,那么对于服务器的压力是很大的,于是就有了基于token的登录状态token在我看来就是根据时间戳,机型号,以及对应的用户id以及其它可以获得的一些信息运用算法生成的一个基本上不会重复的字符串,我们可以将这样一个字符串放到请求头里,然后我们使用这个字符串进行后续的交互就可以了,就可以不用每次都进行账号与密码的核对,减小服务器的压力。

短信验证码

我们也知道短信验证码也是现在app应用中重要的一环,那我们要如何实现这个功能呢?如果是自制简易app项目的话,那我们肯定就要借助一些第三方的sdk,像是阿里云阿,腾讯云阿这种服务平台上的服务,开通服务以后我们要找到对应的api文档然后进行对应的配置,然后就可以实现核对验证码的功能了。

Redis

运行在内存中的数据库,可以方便的存储一些需要用到的键值对,用法也比较简单,然后读写的速度也要比SQL要快很多,我们如果要运营一个项目一味的使用持久化数据库那样带来的压力会很大,所以我们也要考虑到什么东西应该放到内存数据库当中去,所以知道Redis也是很有必要的。

怎样与客户端进行交互?

其实与web端交互是差不了太多的,都是通过url来传递信息,然后获取到参数以后我们返回一个标准形式的json字符串,客户端接收到以后再做其他的操作,现在前后端分离是很明显的,我们后端开发就只需要返回他们需要的数据就足够了

登录与注册两个模块分别实现的大致思路:

这里由于用到的都是SpringBoot中的知识,详细的代码我也就不贴了就给大家一个思路即可有关于我上述说到的验证码、Redis、token后文我会做如何使用的补充

注册:

  • 用户要注册先发送验证码验证
  • 进入验证的接口,验证成功以后返回成功对应的json
  • 然后进行基本信息的填写
  • 提交以后后台进行持久化的保存返回成功

登录:

密码登录

  • 用户首先发出登录请求,传递账号与密码
  • 后台查询数据库进行匹配
  • 如果匹配成功那么生成一个token挂在响应体的头
  • token存入Redis,返回成功的数据
  • 如果匹配不成功则不生成token直接返回失败对应的数据

验证码登录

这个与密码登录思路是大概相同的只不过需要请求一下发送验证码的接口然后再做匹配就可以了,需要注意的是这里也还是需要生成一下token

Redis如何使用?

我们首先需要学完Redis对应的基础知识,然后我们来到SpringBoot的项目中
首先我们要导入Redis的依赖

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

然后来到application.yml(注意这里我习惯于用yml 个人觉得比properties好用一点)配置关于Redis的基本配置

spring:redis:host: "主机ip"port: "Redis的端口号"password: "Redis设置的密码" #这里要注意运行的时候主机中的Redis一定也要是运行状态的

然后我们就可以使用对应的模板类了,这里需要注意一件事我们需要对Redis进行序列化的配置不然会导致我们传递上去的键值对是乱码的形式传递到服务器中去的 Redis如何进行序列化大家可以去查阅相关的资料哦 然后进行Redis模板对象的自定义配置就好了
这里给大家附上一个工具类里面就是对应的方法了

@Component //声明组件
public class RedisUtils {@Autowired //自动注入RedisTemplate<String,Object> redisTemplate;public String get(String key){return (String)redisTemplate.opsForValue().get(key);}public  void set(String key,Object value){redisTemplate.opsForValue().set(key,value);}public boolean set(String key, Object value, long time) {//加入了过期时间的方法try {if (time > 0) {redisTemplate.opsForValue().set(key,value,time,TimeUnit.SECONDS);} else {set(key, value);}return true;} catch (Exception e) {e.printStackTrace();return false;}}}//这里只实现了简单的getset,如果大家要实现drop等功能也可以去查阅资料

token如何使用?

首先为什么要先提到Redis呢?因为我们可以把账号对应的主键以及token字符串以键值对的形式保存到Redis中
token字符串生成时是按照算法加密的,同样我们也可以进行解密我们可以这样配置一个拦截器,拦截请求的时候拿到token然后将token解密拿出用户的uid,然后从Redis中查出对应的token,如果匹配的话那我们就放行,至于为什么不直接使用用户的uid进行存储呢这可能就涉及到了安全方面的问题,可能我们使用用户独立的uid不断的进行操作,可能请求头中每次都携带这用户的uid不是一种安全的做法,所以我们就使用token进行一次加密,这个我了解的也不是很深。
大致思路就是这样下面进入代码的实现:
首先要导入对应的依赖

   <dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt-api</artifactId><version>0.11.2</version></dependency><dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt-impl</artifactId><version>0.11.2</version><scope>runtime</scope></dependency><dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt-jackson</artifactId> <!-- or jjwt-gson if Gson is preferred --><version>0.11.2</version><scope>runtime</scope></dependency>

然后我们要实现生成token对应的工具类

public class TokenUtils {private final String tokenKey="自定义的密钥,字符串的形式";byte[] bytes=tokenKey.getBytes();private final Key key= Keys.hmacShaKeyFor(Decoders.BASE64.decode(tokenKey));private String Token(String userId, Date date) {System.out.println(bytes.length);SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;JwtBuilder builder = Jwts.builder().setIssuedAt(date) // 设置签发时间.setExpiration(new Date(date.getTime() + 1000 * 60 * 60)).claim("userId",String.valueOf(userId) ) // 设置内容,这里很重要因为后面// 需要解析它.signWith(key); String jwt = builder.compact();return jwt;}public String getToken(String userId){Date date=new Date();int nowTime =(int) (date.getTime()/1000);System.out.println(nowTime);String TokenStr=Token(userId,date);return TokenStr;}
}

然后我们要配置拦截器实现使用token验证登录 这里只写一个preHandle方法的重写,具体的配置类以及如何整理到MVC中大家可以去查阅资料

    @AutowiredRedisUtils redisUtils;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {if(request.getRequestURI().contains("/login") ||request.getRequestURI().contains("/register") || request.getRequestURI().contains("/static"))return true;final String headerToken=request.getHeader("token");log.info("获得token:"+headerToken);if(null==headerToken || headerToken.trim().equals("")){response.sendRedirect("/login403");return false;}System.out.println("========================");try{Claims claims= Jwts.parser().setSigningKey("与之前自己设置的key字符串相同") //注意这里要自己改// **如果key过于短 会报错的.parseClaimsJws(headerToken).getBody();String tokenUserId=(String)claims.get("userId");log.info("解析到uid:"+tokenUserId);String RedisToken=redisUtils.get(tokenUserId); //Redis的查询操作if(RedisToken==null){//   request.setAttribute("uid",tokenUserId);response.sendRedirect("/login403"); //重定向到一个返回没有权限的controllerreturn false;}if(!headerToken.equals(RedisToken)){response.sendRedirect("/login403");return false;}return true;}catch (Exception e){System.out.println(e);return false;}}

token的生成应该运用在登录模块中,这里大家按照逻辑来写就好了对应的工具类已经贴上了

如何实现短信服务?

这里我是使用了腾讯云对应的第三方SDK

  • 首先我们需要来到腾讯云的首页然后搜索短信功能开通服务
  • 然后我们跟着快速入门来

在这个地方创建对应的签名与正文,这里注意要严格按照要求不然会不通过的,然后我们要找到它对应的SDK文档

import com.tencentcloudapi.common.Credential;
import com.tencentcloudapi.common.exception.TencentCloudSDKException;//导入可选配置类
import com.tencentcloudapi.common.profile.ClientProfile;
import com.tencentcloudapi.common.profile.HttpProfile;// 导入对应SMS模块的client
import com.tencentcloudapi.sms.v20210111.SmsClient;// 导入要请求接口对应的request response类
import com.tencentcloudapi.sms.v20210111.models.SendSmsRequest;
import com.tencentcloudapi.sms.v20210111.models.SendSmsResponse;/*** Tencent Cloud Sms Sendsms**/
public class SendSms
{public static void main(String[] args){try {/* 必要步骤:* 实例化一个认证对象,入参需要传入腾讯云账户密钥对secretId,secretKey。* 这里采用的是从环境变量读取的方式,需要在环境变量中先设置这两个值。* 你也可以直接在代码中写死密钥对,但是小心不要将代码复制、上传或者分享给他人,* 以免泄露密钥对危及你的财产安全。* SecretId、SecretKey 查询: https://console.cloud.tencent.com/cam/capi */Credential cred = new Credential("secretId", "secretKey");// 实例化一个http选项,可选,没有特殊需求可以跳过HttpProfile httpProfile = new HttpProfile();// 设置代理// httpProfile.setProxyHost("真实代理ip");// httpProfile.setProxyPort(真实代理端口);/* SDK默认使用POST方法。* 如果你一定要使用GET方法,可以在这里设置。GET方法无法处理一些较大的请求 */httpProfile.setReqMethod("POST");/* SDK有默认的超时时间,非必要请不要进行调整* 如有需要请在代码中查阅以获取最新的默认值 */httpProfile.setConnTimeout(60);/* SDK会自动指定域名。通常是不需要特地指定域名的,但是如果你访问的是金融区的服务* 则必须手动指定域名,例如sms的上海金融区域名: sms.ap-shanghai-fsi.tencentcloudapi.com */httpProfile.setEndpoint("sms.tencentcloudapi.com");/* 非必要步骤:* 实例化一个客户端配置对象,可以指定超时时间等配置 */ClientProfile clientProfile = new ClientProfile();/* SDK默认用TC3-HMAC-SHA256进行签名* 非必要请不要修改这个字段 */clientProfile.setSignMethod("HmacSHA256");clientProfile.setHttpProfile(httpProfile);/* 实例化要请求产品(以sms为例)的client对象* 第二个参数是地域信息,可以直接填写字符串ap-guangzhou,支持的地域列表参考 https://cloud.tencent.com/document/api/382/52071#.E5.9C.B0.E5.9F.9F.E5.88.97.E8.A1.A8 */SmsClient client = new SmsClient(cred, "ap-guangzhou",clientProfile);/* 实例化一个请求对象,根据调用的接口和实际情况,可以进一步设置请求参数* 你可以直接查询SDK源码确定接口有哪些属性可以设置* 属性可能是基本类型,也可能引用了另一个数据结构* 推荐使用IDE进行开发,可以方便的跳转查阅各个接口和数据结构的文档说明 */SendSmsRequest req = new SendSmsRequest();/* 填充请求参数,这里request对象的成员变量即对应接口的入参* 你可以通过官网接口文档或跳转到request对象的定义处查看请求参数的定义* 基本类型的设置:* 帮助链接:* 短信控制台: https://console.cloud.tencent.com/smsv2* sms helper: https://cloud.tencent.com/document/product/382/3773 *//* 短信应用ID: 短信SdkAppId在 [短信控制台] 添加应用后生成的实际SdkAppId,示例如1400006666 */String sdkAppId = "1400009099";req.setSmsSdkAppId(sdkAppId);/* 短信签名内容: 使用 UTF-8 编码,必须填写已审核通过的签名,签名信息可登录 [短信控制台] 查看 */String signName = "签名内容";req.setSignName(signName);/* 国际/港澳台短信 SenderId: 国内短信填空,默认未开通,如需开通请联系 [sms helper] */String senderid = "";req.setSenderId(senderid);/* 用户的 session 内容: 可以携带用户侧 ID 等上下文信息,server 会原样返回 */String sessionContext = "xxx";req.setSessionContext(sessionContext);/* 短信号码扩展号: 默认未开通,如需开通请联系 [sms helper] */String extendCode = "";req.setExtendCode(extendCode);/* 模板 ID: 必须填写已审核通过的模板 ID。模板ID可登录 [短信控制台] 查看 */String templateId = "400000";req.setTemplateId(templateId);/* 下发手机号码,采用 E.164 标准,+[国家或地区码][手机号]* 示例如:+8613711112222, 其中前面有一个+号 ,86为国家码,13711112222为手机号,最多不要超过200个手机号 */String[] phoneNumberSet = {"+8621212313123", "+8612345678902", "+8612345678903"};req.setPhoneNumberSet(phoneNumberSet);/* 模板参数: 若无模板参数,则设置为空 */String[] templateParamSet = {"5678"};req.setTemplateParamSet(templateParamSet);/* 通过 client 对象调用 SendSms 方法发起请求。注意请求方法名与请求对象是对应的* 返回的 res 是一个 SendSmsResponse 类的实例,与请求对象对应 */SendSmsResponse res = client.SendSms(req);// 输出json格式的字符串回包System.out.println(SendSmsResponse.toJsonString(res));// 也可以取出单个值,你可以通过官网接口文档或跳转到response对象的定义处查看返回字段的定义System.out.println(res.getRequestId());} catch (TencentCloudSDKException e) {e.printStackTrace();}}
}

我们按照文档进行相应的配置以后,把它封装成对应的工具类,调用对应的函数就可以实现了。
以上就是我对于登录注册进行的难点上的梳理把算是,给大家提供一些思路但是也不是说做的好只不过是把自己学到的东西整理到博客上,如果有更好的解决方案大家也可以积极评论提出

[SpringBoot]使用token 短信验证码 Redis的功能实现基本的登陆注册操作(含Redis token 验证码如何配置)相关推荐

  1. Android 节日短信送祝福(功能篇:1-数据库操作类与自定义ContentProvider)

    首先,还是展示一下部分目录结构:  在节日短信送祝福的功能实现方面,为了能够方便直观展示实现过程,小编我以Java文件为基础,一个一个来展示,免得到时候这个java文件写点,一下又跳到另外一个java ...

  2. Android-节日短信送祝福(功能篇:1-数据库操作类与自定义ContentProvider)

    郑重声明:[慕课网–中级教程]系列的博文均是根据慕课网上的Android开发中级教程相应的视频,在学习之后写的,文中所涉及的源码与视频中的无太大区别,除开个别地方可能有所改善或者加入了自己的想法.如果 ...

  3. SpringBoot 阿里云短信功能使用

    SpringBoot 阿里云短信功能使用 Jar包引用 上工具类 Jar包引用 <!-- 阿里云核心包 --><dependency><groupId>com.al ...

  4. SpringBoot实现阿里短信SDK发送短信,使用MQ监听器

    先 创建sms短信微服务工程,有可能很多工程要使用我们创建的短信服务. 项目环境: SpringBoot SpringClould 步骤: config SmsConfiguration: /*** ...

  5. 榛子云短信验证平台与springboot集成的短信验证

    登录 - 榛子云短信用户系统 (zhenzikj.com) 上面是登录榛子云短信验证平台的入口,此平台的短信大概为3.1分一条短信. 如何与spring boot进行集成呢,我以注册为例来慢慢讲解 1 ...

  6. 利用python实现短信和电话提醒功能

    有时候,我们需要程序帮我们自动检测某些事件的发生 这个需求是广泛存在的 因此,这里整理了利用python实现短信和电话提醒功能的方法 主要需要完成以下4个步骤: - 安装核心库:twilio - 注册 ...

  7. python提醒事件_利用python实现短信和电话提醒功能的例子

    有时候,我们需要程序帮我们自动检测某些事件的发生 这个需求是广泛存在的 因此,这里整理了利用python实现短信和电话提醒功能的方法 主要需要完成以下4个步骤: - 安装核心库:twilio - 注册 ...

  8. 国际短信系统平台后台功能详解-移讯云短信软件

    国际短信系统平台后台功能详解-移讯云短信软件   后台登陆首页显示 ⦁    用户总量记录: 显示总使用户数量 ⦁    待审核账号数量记录: 等待审核的新用户账号数量 ⦁    待审核签名数量记录: ...

  9. 【script】15行Python代码实现免费发送手机短信推送消息功能(twilio)

    实现的功能 通过代码定时给手机推送短信,短信内容可以自定义文字,当然你也可以去别的网站爬取每日心灵鸡汤,天气预报或其它信息进行推送. 本文主要讲如何实现发送短信的功能,全部代码只用15行. 首先贴出实 ...

最新文章

  1. 181920.net用 文字存档
  2. RxJava初级解析(一)
  3. php过滤第一个逗号和最后一个逗号,PHP字符过滤函数去除字符串最后一个逗号(rtrim)...
  4. 数据可视化【一】JavaScript学习
  5. Centos安装nmon软件 ( 测试 )
  6. Qt笔记-窗口程序在任务栏中隐藏及永久置顶
  7. 在计算机中能不能隐藏图片,意想不到的加密方法-怎么把任意文件隐藏在图片里...
  8. Android子线程进度条不显示的问题
  9. NET常出现的三十三种代码(1)
  10. mysql备份 php_详解MYSQL的备份还原(PHP实现)
  11. cc2530设计性实验代码五
  12. 图解设计模式:行为型模式之责任链模式
  13. MySQL忘记密码应当如何重置
  14. 车辆维修管理系统mysql_4S店汽车维修管理系统的分析与设计(JSP+MySQL)
  15. mysql的升序和降序
  16. 费希纳定律的推导过程图解
  17. 数据数值转换factorize和dummy
  18. 夜来风雨声,“网络应用层”知多少?
  19. pcl命名空间:segmentation/extract_clusters.h与segmentation/imp/extract_clusters.hpp
  20. win10怎么将计算机放桌面壁纸,win10电脑系统怎么改变桌面壁纸的大小

热门文章

  1. read函数的 用法
  2. 干掉AlibabaProtect【windows10】
  3. C++【引用】——串讲
  4. vue element upload组件 file-list的动态绑定
  5. python中的“ ... ”
  6. Dracula PRO 2021 零售版
  7. 【转】详细解析电源滤波电容的选取与计算
  8. 百度收购YY直播的“一鱼多吃“
  9. MySQL本地可以连接,远程连接不上的问题
  10. (九五至尊)九大管理体系,五大过程组:软考高级信息系统项目管理师