redis springboot案例分析-mile米乐体育

redis springboot案例分析

这篇文章主要介绍“redis springboot案例分析”,在日常操作中,相信很多人在redis springboot案例分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”redis springboot案例分析”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

一、项目环境

  1. 前端技术栈:vue-cli

  2. 前端软体:webstorm 2020.3

  3. 前端样式: bootstrap

  4. 后端技术栈:springboot

  5. 后端软体:intellij ieda2019

  6. javajdk:1.8

  7. 服务器:阿里云centos 7

  8. 其他:mybatis,redis,mysql,docker,shiro

二、项目演示

  1. 项目源码:shoppingproject01_pub : version6.0

  2. 项目参考:project05;不良人_vue-cli;不良人_redis;不良人_axios;尚硅谷_redis

  3. 项目功能:
    1)邮箱注册登录:
    用户应用邮箱注册点击提交后,网站会给用户发送邮件附带激活码链接,用户点击链接实现账号激活传送门。
    2)短信注册登录:
    用户应用手机号注册时,点击“获取验证码”按钮,手机会接收到网站发送的短信附带验证码。基于redis实现了验证码有效期5分钟,每个手机号只能获取三次短信验证码传送门。
    3)alipay支付:
    通过下载android版支付宝沙箱app用户可通过alipay扫码购买网站上的商品,后台mysql会记录该订单行为传送门,网页展示如图1所示。

图1 商品展示页面

4)用户小分级:
当用户扫码购买年度vip会员后,购买网站上的商品一律半价,后台mysql记录用户角色的变更。
5)用户积分排行榜:
用户购买商品会增加自己的积分,网页展示如图2所示。

图2 排行榜展示页面
  1. 项目中遇到的大坑:
    1)邮件发送功能本地测试通过,服务器端测试bug频出,解决办法。
    2)项目在服务器部署后,无法连接服务器上的redis。解决办法:(1)将redis在服务器部署而非docker;(2)将redis端口改为7000;(3)防火墙active状态下放行服务器和阿里云的7000端口;(4)修改redis.conf文件。
    3)git上传本地源码到gitee,误操作导致本地源码被gitee上的旧代码覆盖,第二天才发现。解决办法:因为在服务器上留有源码的jar包,通过反编译工具jd_gui救回半条命。另外git上传文件参考传送门。

三、主要模块说明

1vue-cli模块说明:

1.1 vue-cli的概述:

1)以前后端分离、单页面web应用(spa)为特点,vue-cli可以创建一个vue项目,这个项目存在脚手架规范。vue-cli的优势如下:
(1) 基于脚手架规范的开发会变得很灵活。
(2) vue-cli基于webpack构建并带有合理的默认配置,打包工具webpack能够聚合单页面和各种开发组件。
(3) vue-cli是一个丰富的官方插件集合,继承了前端生态中最好的工具。

2)安装过程:
(1) 安装webstorm(用于开发),安装node.js,安装vue-cli,安装axios(用于发起跨域请求),引入bootstrap样式。

3)部署过程:

npmrunbuild#在webstorm终端执行,生成dist文件夹dockerpullnginx:1.19.10#不建议vue-cli项目部署到tomcat,因为tomcat属于动态服务器,启动需要java环境,是为了解析动态语言jsp的;像纯静态的就部署到静态服务器nginx上。mkdirhtml#为了做docker容器内外的数据卷映射mvdist/html/dockerrun-p80:80--namenginx01-d-v/root/html/dist/:/usr/share/nginx/htmlnginx:1.19.10#数据卷映射#此时可访问http://120.79.133.235:80/index.html

4)vue-cli开发要点:
(1)在webstorm中,开发过程主要面向src文件,如图3所示:

图3 webstorm目录


[1] 首先掌握路由(router)和组件(components[公有组件],views[私有组件]),组件就是一个个“页面”,组件建立后要向路由进行注册; [2] asserts封装了bootstrap样式,并在main.js中被导入; [3] 为了发送跨域请求,在utils中封装了axios实例,代码如下:

importaxiosfrom'axios'//创建默认实例constinstance=axios.create({baseurl:'http://120.79.133.235:8989/eb',//timeout:10000,});//请求拦截器instance.interceptors.request.use(config=>{console.log("请求拦截器");returnconfig;})//响应拦截器instance.interceptors.response.use(response=>{console.log("响应拦截器");returnresponse;},err=>{console.log("响应出现错误时进入的拦截器");});//暴露instance实例对象exportdefaultinstance;

在各组件中,对于后端的get和post请求方法如下:

//get请求//向后端接口发当前页码,获取当前页面的商品listinstance.get("/item/findallitem?page=" this.page).then(res=>{that.items=res.data.items;that.totalpage=res.data.totalpage;that.page=res.data.page;});//post请求//向后端接口发送当前商品id和用户id,获取商品购买状态instance.post("/order/alipay/callback",{itemid:this.itemid,userid:this.user.id}).then(res=>{if(res.data.code==20000){alert("提示:您已购买该商品");}else{alert("提示:您未购买该商品");}});}

[4] 组件之间的跳转和传值方法如下:

//跳转到mailreg组件this.$router.push({name:"mailreg"});//跳转到item组件,并传递当前商品的idthis.$router.push({path:"/item",query:{itemid:myid}});//item组件接收方法:this.itemid=this.$route.query.itemid;//另外不同组件可以依据token获取登录用户信息,需要用到redis,详见下文

2用户积分排行榜模块说明:
1.1 reids概述:
1) redis是一种基于内存的数据存储nosql;
2) redis支持丰富的数据类型(string, list, set, zset, hash);
3) redis有两种持久化方法: (1)快照(snapshot)存储,也叫rdb持久化,保存当前时刻的数据状态;(2) aof(append only file)存储,将所有redis的写命令记录到日志文件中,redis支持持久化间隔最快也是一秒,所以它是事务不安全的,即是可能丢失数据的。
4)redis的应用场景:
(1) 利用redis字符串完成项目中手机验证码存储的实现。------本项目采用
(2) 利用redis字符串类型完成具有时效性的业务功能,如订单还有40分钟即将关闭。
(3) 利用redis实现分布式集群系统中的session共享。
(4) 利用redis的zset数据类型(可排序set类型:元素 分数)实现排行榜功能。 ------本项目采用
(5) 利用redis完成分布式缓存。 ------本项目实现mysql中数据的缓存
(6) 利用redis存储认证之后的token信息。 ------非常方便,本项目采用。
(7) 利用redis解决分布式集群系统中分布式锁问题。

1.2基于redis实现前端组件从后端获取用户信息
step1:前端login.vue组件中用户输入登录信息提交的接口如下:

//这里调用了后端/user/login接口,获取当前登录用户的token,存入session的localstorage中,在后续网页浏览过程中可随时调取这个tokeninstance.post("/user/login",this.user).then(res=>{if(res.data.state){alert(res.data.msg ",点击确定进入mile米乐体育主页");//前端存储token信息localstorage.setitem("token",res.data.token);that.$router.push({path:"/itemlist"});}else{alert(res.data.msg);that.user={};}});

step2:后端/user/login接口实现如下:

//controller层@postmapping("login")publicmaploginaccount(@requestbodyuseruser,httpsessionsession){returnuserservice.loginaccount(user,session);}//service层//情况3:查询到一个用户时//获取主体对象try{subjectsubject=securityutils.getsubject();subject.login(newusernamepasswordtoken(user.getname(),user.getpassword()));useruserdb=userlistdb.get(0);stringtoken=session.getid();if(userdb.getscore()==null){userdb.setscore(0.0);userdao.updateuserscore(userdb);}redistemplate.opsforvalue().set("token_" token,userdb,30,timeunit.minutes);redistemplate.opsforzset().add("userrank",userdb,userdb.getscore());map.put("token",token);map.put("state",true);map.put("msg","登录成功");returnmap;...

redis整合springboot有两种template,即redistemplate和stringredistemplate。其中stringredistemplate是redistemplate的子类,两个方法基本一致,不同之处在于操作的数据类型不同,redistemplate中的两个泛型都是object,意味着存储的key和value都可以是一个对象,而stringredistemplate的两个泛型都是string,意味着stringredistemplate的key和value都只能是字符串。
在step2中,我将token和数据库中的用户信息userdb绑定在一起存入了redis中,后续前端组件获取登录用户信息的代码如下:

//从localstorage获取tokenlettoken=localstorage.getitem("token");letthat=this;//发送axios请求,根据token获取用户信息instance.get("/user/token?token=" token).then(res=>{that.user=res.data;console.log(that.user);})

后端/user/token的接口如下:

@getmapping({"token"})publicuserfinduser(stringtoken){system.out.println("接收的token信息:" token);return(user)redistemplate.opsforvalue().get("token_" token);}

step3:用户退出登录时,应消除浏览器中对应的token,后端接口代码如下:

//退出登录@deletemapping({"logout"})publicmaplogout(stringtoken){mapmap=newhashmap<>();try{redistemplate.delete("token_" token);subjectsubject=securityutils.getsubject();subject.logout();map.put("state",true);map.put("msg","提示:退出账户成功");returnmap;}catch(exceptione){e.printstacktrace();map.put("state",false);map.put("msg","提示:退出账户失败");returnmap;}}

1.3基于redis的用户积分排行榜实现
mysql中的用户信息如图4所示:

图4 mysql中的用户信息


redis中的userrank如图5所示:

图5 redis中的userrank

step1:当用户登录时,他的首要任务是接入userrank对应的信息,后端代码如下:

if(userdb.getscore()==null){userdb.setscore(0.0);userdao.updateuserscore(userdb);}redistemplate.opsforvalue().set("token_" token,userdb,30,timeunit.minutes);redistemplate.opsforzset().add("userrank",userdb,userdb.getscore());

userdb是数据库中当前登录用户的信息(一定是有的,你注册了,对吧?),若用户首次登录我将他的分数在数据库设置为0.0,之后我在redis的zset中加入这个用户,你知道,set集合不会存储重复key值的元素,因此不会同一个用户出现在userrank中两次。两个template完成了token绑定user,user绑定userrank中score的过程,之后的分数更新过程会反复使用这两个template实现。
step2:当用户信息更新时,相应的与用户信息有关的两个template都要发生变化,代码如下:

//key值序列化redistemplate.setkeyserializer(newstringredisserializer());//由当前用户的token获取当前用户的信息userfirstuser=(user)redistemplate.opsforvalue().get("token_" token);//删除zset中的当前用户redistemplate.opsforzset().remove("userrank",firstuser);//产生新的当前用户(昵称改变)listuserlistdb=this.userdao.selectuserbyname(user.getname());userseconduser=userlistdb.get(0);//更新token中当前用户的信息redistemplate.opsforvalue().set("token_" token,seconduser,30,timeunit.minutes);//产生zset中的当前用户redistemplate.opsforzset().add("userrank",seconduser,seconduser.getscore());

step3:当用户扫码支付时,首次进入的后端controller如下:

//支付单件商品@getmapping("payforitem")publicbyte[]alipay(stringitemid,stringuserid,stringtoken){this.token=token;log.info("itemid=====>" itemid);log.info("userid=====>" userid);payvopayvo=newpayvo();payvo.setitemid(itemid);payvo.setuserid(userid);system.out.println(payvo.getuserid());returnalipayservice.alipay(payvo);}

在alipayservice有一个小型用户分级,即vip用户购物价格减半:

//1:支付的用户stringuserid=payvo.getuserid();//my1:依据用户id查询用户useruser=userservice.selectuserbyid(integer.valueof(userid));//依据商品id查询商品itemitem=itemservice.getitembyid(payvo.getitemid());//my1:依据用户id查询用户if(item==null)returnnull;//2:支付金额stringtempmoney=item.getprice().tostring();stringmoney="";if(user.getrole().equals("normal")){money=tempmoney;}if(user.getrole().equals("vip")){doubletempmoney2=double.valueof(tempmoney)*0.5;money=string.valueof(tempmoney2);}

在payforitem相同文件下,调用了paycommonservice,在这里会实现用户积分更新和用户等级更新:

paycommonservice.payuserpublic(bodyjsonobject,userid,user.getname(),ordernumber,tradeno,"alipay",this.token);

将"vip"这件商品的id设置为“666”,当用户购买该商品时,当前用户更新过程如下:

if(itemid.equals("666")){intmyuserid=integer.valueof(userid);useruserdb=userservice.selectuserbyid(myuserid);//key值序列化this.redistemplate.setkeyserializer(newstringredisserializer());//由当前token获取当前用户信息userfirstuser=(user)redistemplate.opsforvalue().get("token_" token);//由当前用户信息删除当前用户zsetredistemplate.opsforzset().remove("userrank",firstuser);//更新当前用户信息身份userdb.setrole("vip");//更新当前用户新身份的分数userservice.updateuserrole(userdb);listuserlistdb=this.userdao.selectuserbyname(userdb.getname());//获取当前新身份用户的完整信息userseconduser=userlistdb.get(0);//更新当前token对应的当前用户redistemplate.opsforvalue().set("token_" token,seconduser,30,timeunit.minutes);//设置当前用户的zsetredistemplate.opsforzset().add("userrank",seconduser,seconduser.getscore().doublevalue());}

当前用户积分更新过程如下:

//更新当前用户的积分doubletempscore=double.valueof(orderdetail.getprice())*0.3;stringkey1="token_" token;//由当前token获取当前用户useruser=(user)redistemplate.opsforvalue().get(key1);//更新当前用户的zset分数redistemplate.opsforzset().incrementscore("userrank",user,tempscore);//获取当前用户的zset分数doublenewscore=redistemplate.opsforzset().score("userrank",user);//删除当前用户的zset(因为要更新当前用户的信息,将当前用户在数据库中的分数进行同步)redistemplate.opsforzset().remove("userrank",newobject[]{user});user.setscore(newscore);userdao.updateuserscore(user);//更新token对应的当前用户的信息redistemplate.opsforvalue().set(key1,user);//新增当前用户的zsetredistemplate.opsforzset().add("userrank",user,newscore);

到此,关于“redis springboot案例分析”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注恰卡编程网网站,小编会继续努力为大家带来更多实用的文章!

展开全文
内容来源于互联网和用户投稿,文章中一旦含有米乐app官网登录的联系方式务必识别真假,本站仅做信息展示不承担任何相关责任,如有侵权或涉及法律问题请联系米乐app官网登录删除

最新文章

网站地图