1. 需求分析

手机快速登录功能,就是通过短信验证码的方式进行登录。这种方式相对于用户名密码登录方式,用户不需要记忆自己的密码,只需要通过输入手机号并获取验证码就可以完成登录,是目前比较流行的登录方式。

2. 手机快速登录

2.1 页面调整

登录页面为/pages/login.html

2.1.1 发送验证码

为获取验证码按钮绑定事件,并在事件对应的处理函数中校验手机号,如果手机号输入正确则显示30秒倒计时效果并发送ajax请求,发送短信验证码

<div class="input-row"><label>手机号</label><div class="loginInput"><input v-model="loginInfo.telephone" id='account' type="text" placeholder="请输入手机号"><input id="validateCodeButton" @click="sendValidateCode()" type="button" style="font-size: 12px" value="获取验证码"></div></div><div class="input-row"><label>验证码</label><div class="loginInput"><input v-model="loginInfo.validateCode" style="width:80%" id='password' type="text" placeholder="请输入验证码"></div></div>
<script>var vue = new Vue({el:'#app',data:{loginInfo:{}//登录信息},methods:{//发送验证码sendValidateCode(){var telephone = this.loginInfo.telephone;if (!checkTelephone(telephone)) {this.$message.error('请输入正确的手机号');return false;}validateCodeButton = $("#validateCodeButton")[0];clock = window.setInterval(doLoop, 1000); //一秒执行一次axios.post("/validateCode/send4Login.do?telephone=" + telephone).then((response) => {if(!response.data.flag){//验证码发送失败this.$message.error('验证码发送失败,请检查手机号输入是否正确');}});}}
});
</script>

注意:使用计时任务进行30秒倒计时更新

在ValidateCodeController中提供send4Login方法,调用短信服务发送验证码并将验证码保存到redis

/*** 用户进行手机快速登陆,发送验证码* @param telephone* @return*/
@RequestMapping("/send4Login")
public Result send4Login(String telephone) {if (telephone == null) {return new Result(false, MessageConstant.SEND_VALIDATECODE_FAIL);}try {//获取随机验证码Integer code = ValidateCodeUtils.generateValidateCode(6);//发送验证码SMSUtils.sendShortMessage(SMSUtils.VALIDATE_CODE, telephone, code.toString());//将验证码存入RedisjedisPool.getResource().setex(telephone + RedisMessageConstant.SENDTYPE_LOGIN, 500, code.toString());//发送成功通知return new Result(true, MessageConstant.SEND_VALIDATECODE_SUCCESS);} catch (ClientException e) {e.printStackTrace();return new Result(false, MessageConstant.SEND_VALIDATECODE_FAIL);}
}

注意:4在java为for的意思,2为to的意思

2.1.2 提交登录请求

为登录按钮绑定事件

<div class="btn yes-btn"><a @click="login()" href="#">登录</a></div>
//登录
login(){var telephone = this.loginInfo.telephone;if (!checkTelephone(telephone)) {this.$message.error('请输入正确的手机号');return false;}axios.post("/member/login.do",this.loginInfo).then((response) => {if(response.data.flag){//登录成功,跳转到index.htmlwindow.location.href="index.html";}else{//失败,提示失败信息this.$message.error(response.data.message);}});
}

2.2 后台代码

2.2.1 Controller

在health_mobile工程中创建MemberController并提供login方法进行登录检查,处理逻辑为:

  1. 校验用户输入的短信验证码是否正确,如果验证码错误则登录失败

  2. 如果验证码正确,则判断当前用户是否为会员,如果不是会员则自动完成会员注册

  3. 向客户端写入Cookie,内容为用户手机号

  4. 将会员信息保存到Redis,使用手机号作为key,保存时长为30分钟

package com.itheiheihei.controller;import com.alibaba.dubbo.config.annotation.Reference;
import com.alibaba.fastjson.JSON;
import com.itheiheihei.constant.MessageConstant;
import com.itheiheihei.constant.RedisMessageConstant;
import com.itheiheihei.entity.Result;
import com.itheiheihei.pojo.Member;
import com.itheiheihei.service.MemberService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import redis.clients.jedis.JedisPool;import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletResponse;
import java.util.Date;
import java.util.Map;/*** 处理会员相关操作** @author 嘿嘿嘿1212* @version 1.0* @date 2019/10/22 16:29*/
@RestController
@RequestMapping("/member")
public class MemberController {@Autowiredprivate JedisPool jedisPool;@Referenceprivate MemberService memberService;@RequestMapping("/login")public Result check(HttpServletResponse response, @RequestBody Map map) {String validateCode = (String) map.get("validateCode");String telephone = (String) map.get("telephone");//判空if (validateCode == null || telephone == null) {return new Result(false, MessageConstant.TELEPHONE_VALIDATECODE_NOTNULL);}//获取Redis中的验证码String code = jedisPool.getResource().get(telephone + RedisMessageConstant.SENDTYPE_LOGIN);if (validateCode.equals(code)) {//验证码成功Member member = memberService.findByTelephone(telephone);if (member == null) {//自动注册会员member = new Member();member.setRegTime(new Date());member.setPhoneNumber(telephone);memberService.add(member);}//创建Cookie,存入手机号Cookie cookie = new Cookie("login_member_telephone", telephone);cookie.setPath("/");cookie.setMaxAge(60 * 60 * 24 * 30);response.addCookie(cookie);//将会员存入Redis中String jsonMember = JSON.toJSONString(member);jedisPool.getResource().setex(telephone, 60 * 30, jsonMember);return new Result(true, MessageConstant.LOGIN_SUCCESS);} else {//验证码错误return new Result(false, MessageConstant.VALIDATECODE_ERROR);}}
}

2.2.2 服务接口

在MemberService服务接口中提供findByTelephone和add方法

public void add(Member member);public Member findByTelephone(String telephone);

2.2.3 服务实现类

在MemberServiceImpl服务实现类中实现findByTelephone和add方法

package com.itheiheihei.service.impl;import com.alibaba.dubbo.config.annotation.Reference;
import com.alibaba.dubbo.config.annotation.Service;
import com.itheiheihei.dao.MemberDao;
import com.itheiheihei.pojo.Member;
import com.itheiheihei.service.MemberService;
import com.itheiheihei.utils.MD5Utils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.annotation.Transactional;/*** 会员服务** @author 嘿嘿嘿1212* @version 1.0* @date 2019/10/22 18:18*/
@Service(interfaceClass = MemberService.class)
@Transactional
public class MemberServiceImpl implements MemberService {@Autowiredprivate MemberDao memberDao;@Overridepublic Member findByTelephone(String telephone) {return memberDao.findByTelephone(telephone);}@Overridepublic void add(Member member) {String password = member.getPassword();if (password != null) {//使用MD5将密码的明文进行加密password = MD5Utils.md5(password);member.setPassword(password);}memberDao.add(member);}
}

2.2.4 Dao接口

在MemberDao接口中声明findByTelephone和add方法

public Member findByTelephone(String telephone);
public void add(Member member);

2.2.5 Mapper映射文件

在MemberDao.xml映射文件中定义SQL语句

<!--新增会员-->
<insert id="add" parameterType="com.itheiheihei.pojo.Member"><selectKey resultType="java.lang.Integer" order="AFTER" keyProperty="id">SELECT LAST_INSERT_ID()</selectKey>insert into t_member(fileNumber,name,sex,idCard,phoneNumber,regTime,password,email,birthday,remark)values(#{fileNumber},#{name},#{sex},#{idCard},#{phoneNumber},#{regTime},#{password},#{email},#{birthday},#{remark})</insert><!--根据手机号查询会员-->
<select id="findByTelephone" parameterType="string" resultType="com.itheiheihei.pojo.Member">select *from t_memberwhere phoneNumber = #{phoneNumber}
</select>

3. 权限控制

3.1 认证和授权概念

  • 前面我们已经完成了XX健康后台管理系统的部分功能,例如检查项管理、检查组管理、套餐管理、预约设置等。接下来我们需要思考2个问题:

  • 问题1:在生产环境下我们如果不登录后台系统就可以完成这些功能操作吗?

    • 答案显然是否定的,要操作这些功能必须首先登录到系统才可以。
  • 问题2:是不是所有用户,只要登录成功就都可以操作所有功能呢?

    • 答案是否定的,并不是所有的用户都可以操作这些功能。不同的用户可能拥有不同的权限,这就需要进行授权了。

认证:系统提供的用于识别用户身份的功能,通常提供用户名和密码进行登录其实就是在进行认证,认证的目的是让系统知道你是谁

授权:用户认证成功后,需要为用户授权,其实就是指定当前用户可以操作哪些功能

本文章就是要对后台系统进行权限控制,其本质就是对用户进行认证和授权。

3.2 权限模块数据模型

前面已经分析了认证和授权的概念,要实现最终的权限控制,需要有一套表结构支撑:

用户表t_user、权限表t_permission、角色表t_role、菜单表t_menu、用户角色关系表t_user_role、角色权限关系表t_role_permission、角色菜单关系表t_role_menu

表之间关系如下图:

通过上图可以看到,权限模块共涉及到7张表。在这7张表中,角色表起到了至关重要的作用,其处于核心位置,因为用户、权限、菜单都和角色是多对多关系

接下来我们可以分析一下在认证和授权过程中分别会使用到哪些表:

认证过程:只需要用户表就可以了,在用户登录时可以查询用户表t_user进行校验,判断用户输入的用户名和密码是否正确。

授权过程:用户必须完成认证之后才可以进行授权,首先可以根据用户查询其角色,再根据角色查询对应的菜单,这样就确定了用户能够看到哪些菜单。然后再根据用户的角色查询对应的权限,这样就确定了用户拥有哪些权限。所以授权过程会用到上面7张表。

本案例将会使用 Spring Security权限控制框架进行权限控制

XX健康:移动端开发手机验证码快速登陆相关推荐

  1. XX健康:移动端开发-体检预约验证码30秒倒计时短信验证码获取与验证DatePicker日历展示提交预约复杂流程阿里短信工具类

    1. 体检预约流程 用户可以通过如下操作流程进行体检预约: 在移动端首页点击体检预约,页面跳转到套餐列表页面 在套餐列表页面点击要预约的套餐,页面跳转到套餐详情页面 在套餐详情页面点击立即预约,页面跳 ...

  2. XX健康:移动端开发-体检预约设计和实现微信公众号注册阿里短信服务

    1. 移动端开发 1.1 移动端开发方式 随着移动互联网的兴起和手机的普及,目前移动端应用变得愈发重要,成为了各个商家的必争之地.例如,我们可以使用手机购物.支付.打车.玩游戏.订酒店.购票等,以前只 ...

  3. 传智健康_第9章 移动端开发-手机快速登录、权限控制

    传智健康_第9章 移动端开发-手机快速登录.权限控制 文章目录 传智健康_第9章 移动端开发-手机快速登录.权限控制 1. 需求分析 2. 手机快速登录 2.1 页面调整 2.1.1 发送验证码 2. ...

  4. 验证码登录开发----手机验证码登录

    手机验证码登录 需求分析 为了方便用户登录,移动端通常都会提供通过手机验证码登录的功能 手机验证码登录的优点: 方便快捷.无需注册,直接登录 使用短信验证码作为登录凭证,无需记忆密码 安全 登录流程: ...

  5. 手机验证码平台,怎么发送手机验证码,php开发手机验证码短信接口功能

    通过前面的学习, 你已经掌握怎么在腾讯短信平台上设置项目, 添加短信签名, 设置短信模板, 把手机验证码注册页面写好. 今天子恒老师来跟你分享怎么实现发送手机验证码给用户, 注意我们使用的是php开发 ...

  6. 传智健康 ----- 移动端开发 (体检预约)

    1. 移动端开发 1.1 移动端开发方式 随着移动互联网的兴起和手机的普及,目前移动端应用变得愈发重要,成为各个商家的必争之地.例如,我们可以使用手机购物.支付.打车.玩游戏.订酒店.购票等.以前只能 ...

  7. 项目开发过程中如何使用免费开发手机验证码验证功能(详细教程)

    一群热爱技术并且向往优秀的程序猿同学,不喜欢水文,不喜欢贩卖焦虑,只喜欢谈技术,分享的都是技术干货.Talk is cheap. Show me the code 首先声明一下,这个Demo仅限用于开 ...

  8. 移动端开发---手机使用验证码快速登录

    手机快速登录功能,就是通过短信验证码的方式进行登录.这种方式相对于用户名密码 登录方式,用户不需要记忆自己的密码,只需要通过输入手机号并获取验证码就可以完 成登录,是目前比较流行的登录方式. 1:手机 ...

  9. python --- 短信接口开发手机验证码发送

    现在网络环境下,基本上任何网站注册都会验证手机号,已达到防止机器人注册的目的.除此之外短信群发,查询回复,找回密码等相关功能也需要短信验证码功能.那么网站的验证码发送是如何实现的呢?现在我们学习一下( ...

最新文章

  1. android sco通信,android – startBluetoothSco()在ICS上抛出安全异常(BROADCAST_STICKY)
  2. rust(43)-rust语言特点与版本发布
  3. VS code常用插件推荐(总结整理篇)
  4. mvc:default-servlet-handler/作用
  5. linux下软件发布,Linux Kernel 5.12发布下载,附新特性及新功能介绍
  6. Pycharm导入anaconda环境
  7. Python 金融数据可视化(两列数据的提取//分别画//双坐标轴//双图//两种不同的图)...
  8. Inside the Linux Operating System[1]
  9. ethtool查看网卡以及修改网卡配置
  10. Spring 常见注解原理和自定义@interface注解
  11. php表白情话,唯美表白情话短句 八字古风情话
  12. 《保姆教程一》Idea 必装插件,墙裂推荐!!!
  13. 外国人发短信时常用的英文缩写
  14. 产品经理如何进行数据分析?看这一篇文章就够了
  15. 关于Flutter的渠道(channels):master、dev和beta
  16. 【Dubbo】Apache Dubbo 的毕业之旅
  17. 今天收到的QQ礼品卡,差点上当……
  18. 计算机机型分pc机和什么,pc机是什么
  19. 猿如意中的【格式工厂】工具的安装与使用教程,格式转换这个工具就够了
  20. 剑指offer:滑动窗口的最大值(Python)

热门文章

  1. 【知了堂学习笔记】_Java笔试题整理(二)
  2. Calico网络策略原理
  3. 管理理念:如何让员工服从管理
  4. 怎么让员工服从管理_怎样做到让员工服从的管理者
  5. [Oracle]如何获取指定表的约束类型(user_constraints 和user_cons_columns的联系使用)
  6. python自动化测试需要知识_自动化测试需要学习那些知识
  7. Airflow核心源码解读
  8. The Dimpled Manifold Model of Adversarial Examples in Machine Learning 文献阅读
  9. 关于VMWARE的【挂起】与【关机】的区别
  10. java判断字符串是否为数字的几种方式