黑马点评面试话术

文章目录

      • 1.项目介绍
      • 2. 分布式登录功能
        • 2.1 讲讲登录的整个流程
        • 2.2 集群模式session下存储用户信息会有啥问题?
        • 2.3 为什么采用redis存储用户信息和验证码
        • 2.4 redis的存储格式怎么样的?
        • 2.5 为什么采用Hash结构存储用户信息
        • 2.6 为什么采用双拦截器?每个拦截器负责什么功能
      • 3 商品缓存功能
        • 3.1 什么是延时双删?为什么要删两次缓存
      • 4 优惠券秒杀功能
        • 4.1 全局唯一ID是怎么实现的?
        • 4.2 秒杀下单的流程是怎么样的?
        • 4.3 如何解决超卖问题
        • 4.4 如何实现一人一单
        • 4.5 锁在事务内部会存在什么问题
        • 4.6 分布式锁的误删情况了解吗?
        • 4.7 如何实现调用lua
        • 4.8 分布式锁失效问题如何解决
      • 5.秒杀优化
        • 5.1 秒杀优化流程解析
        • 5.2 你使用Jmemter主要是看什么
        • 5.3 你的阻塞队列的东西什么时候提交线程任务呢?
        • 5.4 为什么采用redis的消息队列,而不使用jvm的阻塞队列?
        • 5.5 聊一下redis有哪些消息队列?
      • 6.多用户行为管理
        • 6.1 如何设计点赞排行榜(时间)功能的
        • 6.2 如何实现关注和共同关注
        • 6.3 如何实现附近商铺功能的
        • 6.4 如何实现用户签到功能

1.项目介绍

美食项目主要是一个基于SpringBoot的前后端分离项目,集成了redis,tomcat,mysql。类似于大众点评。实现了短信登录,商户查询缓存,优惠券秒杀,附近的商户,UV统计,用户签到,好友关注,达人探店

2. 分布式登录功能

分布式登录功能:基于Redis+双拦截机制解决集群下session共享问题,利用ThrealLocal存储用户信息

2.1 讲讲登录的整个流程

1.验证码生成:用户提交手机号,点击发送验证码,后端生成验证码,存储(phone,code)到redis,返回验证码到前端
2.登录:提交手机号和验证码,redis中获取验证码校验,校验过了查询用户,存在则UUID 生成token,存储(token, user)到redis,不存在就新建用户先,再存储到redis。返回token信息给前端

2.2 集群模式session下存储用户信息会有啥问题?

tomcat并不共享session内存空间,一台tomcat无法获取存储在另一台tomcat的session数据

2.3 为什么采用redis存储用户信息和验证码

redis基于内存,速度快,可以实现集群共享
可以存储key,value数据结构

2.4 redis的存储格式怎么样的?

验证码:(key, value) —> (手机号,随机六位数字) String结构
用户信息:(key, value)–> (token, 用户信息) Hash结构

2.5 为什么采用Hash结构存储用户信息

内存占用少,存取操作方便。针对某个字段可以做增删改操作。
String类型需要全部删除

2.6 为什么采用双拦截器?每个拦截器负责什么功能

第一个拦截器:判断token是否存在,存在找到用户信息,存入threadlocal。并刷新token有效期,然后放行
判断threadlocal是否有值,有则放行,无则拦截

3 商品缓存功能

采用Redis实现商品毫秒级响应,采用延时双删策略保证数据的一致性。

3.1 什么是延时双删?为什么要删两次缓存

延时双删是为了解决数据库数据和缓存数据不一致的问题。先删除缓存,再更新数据库,延迟一会再删除缓存。删两次缓存时因为并发环境下,可能数据库还没更新成功,请求来了读取到了数据库的旧数据,回填到redis。所以需要延时一会,再删一次缓存。

4 优惠券秒杀功能

基于Redis实现全局唯一订单ID,CAS解决库存超卖问题,Redission分布式锁解决一人一单并发安全问题

4.1 全局唯一ID是怎么实现的?

符号位+时间戳+计数器,应用到了String中的incrby功能
其中key采用每天生成不同的key,方便统计每天的订单数量

4.2 秒杀下单的流程是怎么样的?

提交优惠券ID,查询优惠券信息,判断秒杀是否开始,判断库存是否充足,查询订单,校验一人一单,减库存,均满足创建订单,返回订单ID即可

4.3 如何解决超卖问题

使用CAS乐观锁,在扣减库存的时候判断库存是否大于0.但是本质还是用到了数据库中的行锁吧,修改数据加了写锁。这里锁的是库存

4.4 如何实现一人一单

在扣减库存之前,到订单表查询用户ID和优惠券ID是否存在存在,存在就拒绝后续动作。这里要加悲观锁,锁用户ID,注意这里通过userId.toString.intern()获取字符串常量池的规范表示,值相同的地址相同,可以锁住,因为toString方法都会底层是新new了一个对象

注意:这里锁的是用户ID,要保证事务提交了,锁才能释放

4.5 锁在事务内部会存在什么问题

锁释放了,但是事务未提交,其他线程进来,查询认为数据库数据未更新,因此可能出现并发安全问题。一般要保证提交事务之后再释放锁

4.6 分布式锁的误删情况了解吗?

线程1,拿到锁,但是业务执行时间太久了,锁过期自动释放。线程2获得redis锁,执行业务,此时线程1业务执行完了,又执行释放锁动作,错误把线程2的锁释放了。
解决方案,释放锁的时候判断UUID+线程ID是否是自己。因为不同集群下线程ID可能一致
这里还存在什么问题吗?
判断锁和释放锁之间执行Full GC 业务代码阻塞,锁超时释放。GC结束,线程2获取锁成功,然后线程1释放锁,又把线程2的锁释放了
所以需要lua脚本实现判断锁和释放锁的原子性

4.7 如何实现调用lua

使用redistemplate的execute(lua,key,value)

4.8 分布式锁失效问题如何解决

采用多个redis节点作为主节点,称之为联锁。只要所有节点获取锁成功,才算成功

5.秒杀优化

使用lua脚本实现库存和用户下单资格判断,采用Stream消费者组实现消息队列存放下单信息,通过单例线程池实现异步下单入库操作,优化接口平均响应时间从 2500ms 降低到 110ms,QPS 提升显著

5.1 秒杀优化流程解析

把判断秒杀库存和校验一人一单功能放到redis做,输出优惠券ID,用户ID,订单ID到阻塞队列中,开启线程异步读取线程信息,完成下单。
在这里插入图片描述
在这里插入图片描述

5.2 你使用Jmemter主要是看什么

1.平均值:所有请求响应时间的平均值
2.吞吐量:每秒处理成功的请求,本机达到(1500)

5.3 你的阻塞队列的东西什么时候提交线程任务呢?

类初始化之后就提交任务,使用到了PostConstruct注解(和bean的生命周期可以联动)

5.4 为什么采用redis的消息队列,而不使用jvm的阻塞队列?

jvm内存限制,内存故障,消息丢失
redi实现的消息队列可以实现持久化机制,并且不受jvm内存限制

5.5 聊一下redis有哪些消息队列?

List,PubSub和Stream
本项目采用Stream消息队列的原因:支持消费者组,组内竞争消息,避免重复消费,又消费确认机制,保证消息至少被消费一次。吞吐量达到1600

6.多用户行为管理

SortedSet实现笔记点赞排行榜、Set集合管理用户共同关注、Geo实现附近商铺功能,BitMap实现用户签到

6.1 如何设计点赞排行榜(时间)功能的

redis的sortedset实现,key为笔记ID,value为点赞的用户及时间。可以根据时间排序返回

6.2 如何实现关注和共同关注

使用redis的set数据结构,每个用户存储他关注的所有人信息。
要求两个用户的话,拿出来用SInter求交集就行

6.3 如何实现附近商铺功能的

使用到了GEO功能。GEOADD把商铺根据类型作为key,经纬度作为value存入redis.
GEOSearch按照半径返回附近店铺的坐标信息。

6.4 如何实现用户签到功能

使用BitMap,相对于mysql存储记录,节省了大量的内存空间
主要是他的每一位对应的0和1,只要签到了,就记录为1即可
存储的key为用户+月份。只要4个字节即可

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

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

相关文章

MTK APEX测光系统中各变量具体的计算方式探究

目录 一、APEX测光系统介绍 二、MTK测光系统实例介绍 三、关于测光系统的一些疑问 一、APEX测光系统介绍 详细内容可以参考; AE(自动曝光)系统简介

K8S的基本概念

Kubernetes是一个开源的容器编排部署管理平台,用于管理云平台中多个主机上的容器化应用。Kubernetes的目标是让部署容器化的应用简单并且高效,Kubernetes提供了应用部署、规划、更新、维护的一种机制。 对应用开发者而言,可以把Kubernetes看成一个集群操作系统。Kubernetes…

NLP学习路线图(三十四): 命名实体识别(NER)

一、命名实体识别(NER)是什么? 命名实体识别(Named Entity Recognition, NER)是自然语言处理中的一项关键序列标注任务。其核心目标是从非结构化的文本中自动识别出特定类别的名词性短语,并将其归类到预定义的类别中。 核心目标:找到文本中提到的命名实体,并分类。 典…

大三自学笔记:探索Hyperlane框架的心路历程

## Day 1:初识 Hyperlane 在 GitHub 上发现了 Hyperlane 这个 Rust HTTP 框架,立刻被它的性能数据吸引。官方文档写着: > "hyperlane 是一个高性能且轻量级的 Rust HTTP 框架,设计目标是简化现代 Web 服务的开发&#xff…

Java大厂面试真题:谢飞机的技术挑战

Java大厂面试真题:谢飞机的技术挑战 场景一:电商场景 面试官:在电商项目中,我们通常需要处理大量的并发请求。请谈谈你对JVM调优的理解。 谢飞机:嗯,JVM调优主要是为了提高程序的性能和稳定性。比如&…

【Docker管理工具】安装容器管理工具Oxker

【Docker管理工具】安装Oxker容器管理工具 一、Oxker介绍1.1 Oxker简介1.2 Oxker功能1.3 Docker介绍 二、本地环境介绍2.1 本地环境规划2.2 本次实践介绍 三、本地环境检查3.1 检查Docker服务状态3.2 检查Docker版本3.3 检查docker compose 版本 四、下载Oxker镜像五、安装Oxke…

产品成本分析怎么做?从0到1搭建全生命周期分析框架!

目录 一、为什么要做产品全生命周期成本分析? 1.资源再分配 2.动态成本校准 3.战略决策支持 二、产品成本分析思路 1.建立全生命周期成本追踪 2.联动分析关键指标 3.定位问题产品线 4.资源效率四象限分配 三、产品成本分析指标 1.分投入成本:…

机器学习与深度学习20-数学优化

目录 前文回顾1.梯度下降的基本原理2.什么是损失函数?3.随机梯度下降和小批量梯度下降4.什么是学习率5.优化算法中的收敛性6.常用的数学优化算法 前文回顾 上一篇文章链接:地址 1.梯度下降的基本原理 梯度下降(Gradient Descent&#xff0…

Photoshop 2025 性能配置全攻略:硬件选购与软件优化指南

一、硬件配置核心建议 根据Adobe官方要求及实测反馈,Photoshop 2025对硬件的需求侧重CPU、内存和存储,显卡需求相对宽松,但特定功能(如AI滤镜、3D渲染)需关注显卡性能。 硬件类别最低配置推荐配置(流畅运…

华为云Flexus+DeepSeek征文 | 华为云ModelArts Studio快速上手:DeepSeek-R1-0528商用服务的开通与使用

华为云FlexusDeepSeek征文 | 华为云ModelArts Studio快速上手:DeepSeek-R1-0528商用服务的开通与使用 引言一、ModelArts Studio平台介绍华为云ModelArts Studio简介ModelArts Studio主要特点 二、开通DeepSeek-R1-0528商用服务访问ModelArts Studio控制台DeepSeek-…

day53 神经网络调参指南

目录 一、引言 二、权重初始化:为何如此重要? (一)随机种子:确保实验的可重复性 (二)权重初始化的重要性 1. 神经网络的对称性问题 2. 避免梯度消失和梯度爆炸 (三&#xff0…

【大模型02---Megatron-LM】

文章目录 Megatron-LM数据并行模型并行张量并行流水线并行 3D并行 Megatron-LM Megatron是当前大模型训练时经常使用的一种分布式并行框架,它通过采用DP,TP,PP等来加速模型的训练,反正就是一个字,好。 大模型在训练的时候,显存占…

魔百和网络机顶盒CM211-1硬件解析

先来个正面照 背面照 核芯 无线网卡 支持WiFi与蓝牙 硬盘 正面内存与背面内存

Kratos 与Golang Cms的关系

Kratos 与 Golang CMS 的关系 Kratos 是 Bilibili 开源的一款轻量级 Go 语言微服务框架,专注于构建高性能、可扩展的后端服务。虽然它本身并不是一个完整的 CMS(内容管理系统),但它可以用于开发 CMS 系统的后端或 API 服务。 我们的目标是提供全面的微服务开发技术。基于…

在vue3+vite中给 Video视频 添加字幕

Video视频 添加字幕 方式一: 使用 track标签template标签中css样式修改方式二:直接读取.vtt文件方式一: 使用 track标签 参考1:https://blog.csdn.net/weixin_42321819/article/details/112442773 参考2:https://blog.csdn.net/foren_whb/article/details/80810552 template标…

UE4手动实现billboard效果让物体始终面向相机正面

一个很简单的需求,但在网上竟然没查到。首先不能用FindLookAtRotation,因为这是用location算的,是让物体朝向相机的方向,而不是朝向相机的正面。区别如下图所示: 然后想用billboard component,不过这个原生…

在阿里云上搭建n8n

0.安装docker 0.1 删除Docker相关源 #删除Docker相关源 sudo rm -f /etc/yum.repos.d/docker*.repo #卸载Docker和相关的软件包 sudo dnf -y remove \ docker-ce \ containerd.io \ docker-ce-rootless-extras \ docker-buildx-plugin \ docker-ce-cli \ docker-compose-plug…

Qt中的OpenGL (4)[纹理]

文章说明 本文是学习OpenGL的笔记,主要参考大神JoeyDeVries的LearnOpenGL第六课《纹理》,并将教程中的代码基于Qt进行实现。 学习目标 掌握纹理基本知识掌握纹理贴图目录结构 |- |-- HelloTextures|--- hello_textures.cpp|--- hello_textures.h|--- main.cpp|--- CMakeLi…

【Java多线程从青铜到王者】阻塞队列(十)

阻塞队列 阻塞队列也是一种队列,先进的先出 阻塞队列就是对普通的队列做出的拓展 阻塞队列的特性 1.线程安全的,我们普通的队列值线程不安全的 2.具有阻塞的特性: a》如果针对一个已经满了的队列进行如队列操作的话,入队列操作就…

Python打卡第52天

浙大疏锦行 作业: 对于day41的简单cnn,看看是否可以借助调参指南进一步提高精度。 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms from torch.utils.data import DataLoader import ma…