在线教育-谷粒学院学习笔记(十)

news2024/11/16 23:37:39

文章目录

    • 1 介绍
    • 2 登录业务流程
    • 3 JWT令牌
    • 4 阿里云短信服务
    • 5 登录功能
    • 6 注册功能
    • 7 根据token获取用户信息
    • 8 整合首页登录和注册

1 介绍

  • 登录实现流程

  • 注册接口

    • 整合JWT
    • 整合阿里云短信服务
  • 登录接口

  • 注册、登录的前端实现

2 登录业务流程

  • 单一服务器模式

    • 使用session对象实现

      • 登录成功后,将用户数据放到session

      • 登录时,判断是否登录,从session获取数据,实现登录

      • session.setAttritute("user", user);
        session.getAttribute("user");
        
      • 性能压力,无法扩展

  • 集群部署

    • 分布式
    • 单点登录
  • SSO单点登录模式 single sign on

    • 三种常见方式

      1、session广播机制实现(基本不用了)

      • session复制

      2、使用cookie+redis 实现

      • 1.在项目中任何一个模块进行登录,登录后将数据放到两个地方
        • redis:在key:生成唯一随机值(ip、id等),在value:用户数据
        • cookie:把redis里面生成的key值放到cookie中
      • 2.访问项目中其他模块,发送请求时带着cookie进行发送,获取cookie值后
        • 把获取到的cookie值放入redis进行查询,根据key查询,如果能查到数据,就是登录

      3、使用token实现

      • 令牌

      • token:按照一定的规则生成字符串,字符串可以包含用户信息 自包含令牌

      • 1.在项目某个模块进行登录,登录之后,按照规则生成字符串,将用户包含在字符串里,最后返回字符串

        • 把字符串通过cookie返回
        • 把字符串通过地址栏返回
      • 2.再访问项目其他模块,每次访问在地址栏带着生成的字符串,在访问模块里获取地址栏字符串,根据字符串获取用户信息。如果可以获取到,就是登录

session默认过期时间:30min

session、redis、token都可以设置过期时间

3 JWT令牌

  • JWT

    • token是按照一定规则生成字符串,包含用户信息

    • 规则是怎么样的,不一定

    • 一般采用通用的规则,官方规则JWT

    • JTW就是规定好的规则,使用JWT可以直接生成字符串,包含用户信息

  • JWT生成的字符串包含三部分

    • 1 jwt头信息
    • 2 有效载荷,包含主体信息(用户信息)
    • 3 签名哈希(防伪标志)

1 在common中引入依赖,复制jwt工具类

common/common_utils/pom.xml

<dependencies>
    <!-- JWT-->
    <dependency>
        <groupId>io.jsonwebtoken</groupId>
        <artifactId>jjwt</artifactId>
    </dependency>
</dependencies>

JwtUtils.java

4 阿里云短信服务

整合阿里云短信服务,注册时发生验证码

1 在service中创建子模块service_msm

2 创建包结构,创建controller、service

3 阿里云平台操作

  • 开通阿里云短信服务
  • 进入控制台,点击国内消息,申请两个内容:签名管理、模板管理
  • 申请模板管理
  • 申请签名管理
  • 注:现在有测试版,可以直接使用测试版,不申请

在这里插入图片描述

需求:验证码五分钟内有效

使用Redis解决验证码有效时间

controller

@RestController
@CrossOrigin
@RequestMapping("/edumsm/msm")
public class MsmController {
    @Autowired
    private MsmService msmService;
    @Autowired
    private RedisTemplate<String, String> redisTemplate;

    // 发送短信
    @GetMapping("send/{phone}")
    public R sendMsm(@PathVariable String phone) {
        // 1. 先从redis中获取验证码,如果能获取直接返回
        String code = redisTemplate.opsForValue().get(phone);
        if (! StringUtils.isEmpty(code)) {
            return R.ok();
        }

        // 2. 如果获取不到,再进行阿里云发送
        // 生成随机的值,传递给阿里云发送
        code = RandomUtil.getFourBitRandom();
        Map<String, Object> param = new HashMap<>();
        param.put("code", code);
        // 调用service发送短信的方法
        boolean isSend = msmService.send(param, phone);
        if (isSend) {
            // 发送成功,把发送成功验证码放到redis里,并设置有效时间
            redisTemplate.opsForValue().set(phone, code, 5, TimeUnit.MINUTES);
            return R.ok();
        } else {
            return R.error().message("短信发送失败");
        }
    }
}

service 固定写法,修改自己的参数

@Service
public class MsmServiceImpl implements MsmService {
    // 发送短信
    @Override
    public boolean send(Map<String, Object> param, String phone) {
        if (StringUtils.isEmpty(phone)) return false;

        DefaultProfile profile = DefaultProfile.getProfile("default", "accessKeyId", "secret");
        IAcsClient client = new DefaultAcsClient(profile);

        // 设置相关固定参数
        CommonRequest request = new CommonRequest();
        request.setMethod(MethodType.POST);
        request.setDomain("dysmsapi.aliyuncs.com");
        request.setVersion("2017-05-25");
        request.setAction("SendSms");

        // 设置发送相关参数
        request.putQueryParameter("PhoneNumbers", phone); // 手机号
        request.putQueryParameter("SignName", "阿里云短信测试"); // 签名名称
        request.putQueryParameter("TemplateCode", "SMS_154950909");
        request.putQueryParameter("TemplateParam", JSONObject.toJSONString(param)); // 验证码
        // 最终发送
        try {
            CommonResponse response = client.getCommonResponse(request);
            boolean success = response.getHttpResponse().isSuccess();
            return success;
        } catch (Exception e) {
            e.printStackTrace();
            return false;
        }
    }
}

5 登录功能

1 在service下创建子模块 service_ucenter

2 创建用户表,使用代码生成器生成相关代码

guli_ucenter.sql

ucenter_member

3 配置文件application.properties

4 启动类 UcenterApplication

@SpringBootApplication
@ComponentScan("com.mys")
@MapperScan("com.mys.educenter.mapper")
public class UcenterApplication {
    public static void main(String[] args) {
        SpringApplication.run(UcenterApplication.class);
    }
}

5 controller 、service

controller

// 登录
@GetMapping("login")
public R loginUser(@RequestBody UcenterMember member) {
    // 使用service实现登录,返回token值(jwt生成)
    String token = memberService.login(member);
    return R.ok().data("token", token);
}

service

// 登录
@Override
public String login(UcenterMember member) {
    // 获取登录的手机号和密码
    String mobile = member.getMobile();
    String password = member.getPassword();
    // 手机号和密码非空判断
    if (StringUtils.isEmpty(mobile) || StringUtils.isEmpty(password)) {
        throw new GuliException(20001, "登录失败");
    }
    // 判断手机号是否正确
    QueryWrapper<UcenterMember> wrapper = new QueryWrapper<>();
    wrapper.eq("mobile", mobile);
    UcenterMember mobileNumber = baseMapper.selectOne(wrapper);
    // 判断查询对象是否为空
    if (mobileNumber == null) { // 没有这个手机号
        throw new GuliException(20001, "登录失败");
    }
    // 判断密码
    if (! password.equals(mobileNumber.getPassword())) {
        throw new GuliException(20001, "登录失败");
    }
    // 判断用户是否禁用
    if (mobileNumber.getIsDisabled()) {
        throw new GuliException(20001, "登录失败");
    }

    // 登录成功,使用jwt生成token字符串
    String jwtToken = JwtUtils.getJwtToken(mobileNumber.getId(), mobileNumber.getNickname());
    return jwtToken;
}

测试

6 注册功能

1 创建实体类,封装注册数据,包含验证码属性

entity.vo.RegisterVo

@Data
public class RegisterVo {
    @ApiModelProperty(value = "昵称")
    private String nickname;
    @ApiModelProperty(value = "手机号")
    private String mobile;
    @ApiModelProperty(value = "密码")
    private String password;
    @ApiModelProperty(value = "验证码")
    private String code;
}

2 在controller中创建注册方法

// 注册
@PostMapping("register")
public R registerUser(@RequestBody RegisterVo registerVo) {
    memberService.register(registerVo);
    return R.ok();
}

3 service

// 2 非空判断
    if (StringUtils.isEmpty(mobile) || StringUtils.isEmpty(code) ||
            StringUtils.isEmpty(nickname) || StringUtils.isEmpty(password)) {
        throw new GuliException(20001, "注册失败");
    }

    // 判断验证码
    // 先获取redis中的验证码
    String redisCode = redisTemplate.opsForValue().get(mobile);
    if (code.equals(redisCode)) {
        throw new GuliException(20001, "注册失败");
    }

    // 相同判断:手机号不能重复
    QueryWrapper<UcenterMember> wrapper = new QueryWrapper<>();
    wrapper.eq("mobile", mobile);
    Integer count = baseMapper.selectCount(wrapper);
    if (count > 0) {
        throw new GuliException(20001, "注册失败");
    }

    // 添加数据到数据库
    UcenterMember member = new UcenterMember();
    member.setMobile(mobile);
    member.setNickname(nickname);
    member.setPassword(MD5.encrypt(password)); // 密码加密后存储
    member.setIsDisabled(false); // 用户不禁用
    member.setAvatar("http://thirdwx.qlogo.cn/mmopen/vi_32/DYAIOgq83eoj0hHXhgJNOTSOFsS4uZs8x1ConecaVOB8eIl115xmJZcT4oCicvia7wMEufibKtTLqiaJeanU2Lpg3w/132");
    baseMapper.insert(member);
}

4 swagger测试

成功

7 根据token获取用户信息

// 根据token获取用户信息
@GetMapping("getMemberInfo")
public R getMemberInfo(HttpServletRequest request) {
    // 调用jwt工具类方法,根据request对象,获取头信息,返回用户id
    String memberId = JwtUtils.getMemberIdByJwtToken(request);
    // 查询数据库,根据用户id,获取用户信息
    UcenterMember member = memberService.getById(memberId);
    return R.ok().data("userInfo", member);
}

8 整合首页登录和注册

1 安装插件

npm install element-ui

npm install vue-qriously (微信支付)

2 在nuxt环境中使用element-ui插件

import Vue from 'vue'
import VueAwesomeSwiper from 'vue-awesome-swiper/dist/ssr'
import VueQriously from 'vue-qriously'
import ElementUI from 'element-ui'
import 'element-ui/lib/theme-chalk/index.css'
Vue.use(VueAwesomeSwiper)
Vue.use(VueQriously)
Vue.use(ElementUI)

3 整合页面

  • 在layouts文件夹创建登录注册页面布局sign.vue
  • 修改登录和注册超链接地址 在default.vue
  • 在pages创建register.vue login.vue,复制页面代码

4 整合注册前端

  • api中创建register.js,注册接口相关代码

    // 根据手机号发送验证码
    sendCode(phone) {
      return request({
        url: `/edumsm/msm/send/${phone}`,
        methods: 'post'
      })
    },
    // 注册
    registerMember(formItem) {
      return request({
        url: `/educenter/member/register`,
        methods: 'post',
        data: formItem
      })
    }
    
  • 在页面中调用

    js定时器方法设计倒计时:setInterval(“”, 60)

    格式校验:

    :rules="[{ required: true, message: '请输入手机号码', trigger: 'blur' },{validator:checkPhone, trigger: 'blur'}]
    

    检查nginx端口是否正确,停掉并重启 nginx.exe -s stop

跨域问题

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/184547.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

仿牛客论坛项目(下)

代码仓库:https://gitee.com/qiuyusy/community 仿牛客论坛项目15.kafka1.阻塞队列2.Kafka入门简介术语解释下载配置命令3.Spring整合Kafka引入依赖配置代码16.系统通知(Kafka)发送系统通知功能(点赞关注评论)1.编写Kafka消息队列事件Event实体类2.编写Kafka生产者3.编写Kafka消…

Spring 源码编绎

本示例基于 spring-framework-5.2.22.RELEASE GradleWapper jdk1.8.0_131编译# 环境准备mavenjdk8idea# 源码下载进入https://github.com/spring-projects/spring-frameworkSpring的源码是发布在github上面的下载最新版发布版源码不要太纠结版本区别&#xff0c;无需刻意保证…

BM5 合并k个已排序的链表

目录 描述 示例1 示例2 思路&#xff1a; 代码&#xff1a; 描述 合并 k 个升序的链表并将结果作为一个升序的链表返回其头节点。 示例1 输入&#xff1a;[{1,2,3},{4,5,6,7}] 返回值&#xff1a;{1,2,3,4,5,6,7} 示例2 输入&#xff1a;[{1,2},{1,4,5},{6}] 返回值…

Linux常用命令——rmmod命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) rmmod 从运行的内核中移除指定的内核模块 补充说明 rmmod命令用于从当前运行的内核中移除指定的内核模块。执行rmmod指令&#xff0c;可删除不需要的模块。Linux操作系统的核心具有模块化的特性&#xff0c;应…

Elasticsearch(六)--ES文档的操作(中)---修改文档

一、前言 上篇文章我们了解了ES的插入和批量插入文档的操作&#xff0c;分别通过ES的kibana客户端以及Java高级Rest客户端进行学习&#xff0c;那么本篇则进入到对文档的修改操作&#xff0c;同新增文档&#xff0c;也有更新单条文档和批量更新文档操作&#xff0c;但还多出一…

Day873.普通索引唯一索引的选择 -MySQL实战

普通索引&唯一索引的选择 Hi&#xff0c;我是阿昌&#xff0c;今天学习记录的是关于普通索引&唯一索引的选择的内容。 假设你在维护一个市民系统&#xff0c;每个人都有一个唯一的身份证号&#xff0c;而且业务代码已经保证了不会写入两个重复的身份证号。 如果市民…

Java基础项目实战--大学生求职招聘信息网站系统

Java基础项目实战–大学生求职招聘信息网站系统 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言 文末获取源码联系…

基于php的婚纱商城管理系统

摘要网络技术给生活带来了十分的便利。所以把婚纱商城管理系统与现在网络相结合。在婚纱商城发展的整个过程中&#xff0c;婚纱信息管理担负着最重要的角色。为满足如今日益复杂的管理需求&#xff0c;各类管理系统程序也在不断改进。本课题所设计的婚纱商城管理系统&#xff0…

自动驾驶介绍、应用、前景

自动驾驶介绍、应用、前景1 介绍1.1 定义1.2 作用1.3 发展历程1.4 分类23年初竞争格局1.5 顾虑1.6 前景2 产业链现状2.1 芯片2.2 仿真3 技术路线3.1 是否交互3.1.1 单车智能3.1.2 车路协同3.2 主传感器区分3.2.1 纯视觉3.2.2 混合传感器3.3 前装还是后装3.3.1 前装3.3.2 后装4 …

基于java SSM医药住院管理系统设计和实现

基于java SSM医药住院管理系统设计和实现 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言 文末获取源码联系方式 …

宇隆光电冲刺上交所上市:毛利率持续下滑,收入极其依赖京东方

近日&#xff0c;重庆宇隆光电科技股份有限公司&#xff08;下称“宇隆光电”&#xff09;预披露招股书&#xff0c;准备在上海证券交易所主板上市。 本次冲刺上市&#xff0c;宇隆光电计划募资15亿元&#xff0c;其中7亿元用于OLED控制板及液晶模组控制板和精密模切生产基地项…

Spring Cloud_服务监控hystrixDashboard

目录一、概述二、仪表盘90011.新建Module2.POM3.YML4.主函数5.Provider微服务提供类都需要监控依赖配置6.启动仪表盘三、断路器演示1.修改cloud-provider-hystrix-payment80012.监控测试代码链接 https://github.com/lidonglin-bit/cloud 一、概述 除了隔离依赖服务的调用以外…

背景图片大小设置 解决背景图多张铺满盒子 背景图和背景颜色混用

目录更多的样式透明度鼠标样式&#xff1a;cursor盒子隐藏背景图和img元素的区别涉及的css属性1. background-image:url(“”)2. background-repeat3. background-size解决图片多张铺满盒子的问题4. background-position5. background-attachment6. 背景图和背景颜色混用7. 速写…

Jackson使用详细介绍

Jackson使用详细介绍一 . Jackson 介绍二. Jackson Maven 依赖三. ObjectMapper 对象映射器四. Jackson JSON 基本操作1. Jackson JSON 序列化2. Jackson JSON 反序列化3. JSON 转 List4. JSON 转 Map5. Jackson 忽略字段6. Jackson 日期格式化Date 类型LocalDateTime 类型时间…

动态内存分配

目录 一、内存使用方式 &#xff08;一&#xff09;一个c/c编译的程序占用的内存分为以下几个部分 二、malloc &#xff08;一&#xff09;malloc 1. 举例&#xff1a;malloc(4) 2. 如何理解malloc(size(Var_T)*N) 3. 举例 &#xff08;二&#xff09;静态、全局指针…

【数据结构与算法】队列-模拟实现队列以及设计循环队列

文章目录队列的概念链表实现栈设计循环队列总结队列的概念 队列是一种特殊的线性表&#xff0c;特殊之处在于它只允许在表的前端&#xff08;front&#xff09;进行删除操作&#xff0c;而在表的后端&#xff08;rear&#xff09;进行插入操作&#xff0c;和栈一样&#xff0c;…

图的搜索(DFS、BFS)

图的搜索&#xff08;图的遍历&#xff09;是指从图的任一顶点出发&#xff0c;访问图的所有顶点&#xff0c;且每个顶点只访问一次。 深度优先搜索 DFS概念&#xff1a; 深度优先搜索 (Depth-First Search&#xff0c;DFS)是从某个顶点v1出发对图进行搜素&#xff0c;每一步…

第八章 面向对象编程(中级)

一、访问修饰符&#xff08;P279&#xff09; 1. 基本介绍 java提供四种访问控制修饰符号&#xff0c;用于控制方法和属性&#xff08;成员变量&#xff09;的访问权限&#xff08;范围&#xff09;&#xff1a; &#xff08;1&#xff09;公开级别&#xff1a;用 public 修饰…

2018-NIPS-owards Sparse Hierarchical Graph Classifiers

2018-NIPS-owards Sparse Hierarchical Graph Classifiers Paper: https://arxiv.org/abs/1811.01287 Code: 对稀疏分类分级图 作者提出以往的图分类方法中通常使用单个全局池化步骤来聚合节点特征或手动设计的固定启发式算法&#xff0c;这样做会丢失信息&#xff0c;所以将…

ABB机器人系统输入输出信号System Input和Output详解(二)

ABB机器人系统输入输出信号System Input和Output详解(二) 上一次和大家分享了系统输入信号System Input相关的内容,具体可参考以下链接中的内容: ABB机器人系统输入输出信号System Input和Output详解(一) 本次和大家分享系统输出信号的相关内容: System Output类型: 可…