1.数据库表你是怎么设计的?
分为员工表、用户表、分类表、菜品表、套餐表、套餐菜品对应表、口味表、购物车表、订单表、订单详细表、地址表。
为什么用逻辑外键,而不用数据库自带的外键?
数据库通过外键来保证数据的完整性一致性,外键的缺点--对海量数据,性能低,因而使用逻辑外键来保证数据的完整性和一致性,当然,如果是对安全性要求高以及对性能要求不高的场景,可以使用外键 这样更安全。
2.nginx是起到什么作用?
nginx 反向代理,就是将前端发送的动态请求由 nginx 转发到后端服务器,反向代理的好处:
-
提高访问速度
因为nginx本身可以进行缓存,如果访问的同一接口,并且做了数据缓存,nginx就直接可把数据返回,不需要真正地访问服务端,从而提高访问速度。
-
进行负载均衡
所谓负载均衡,就是把大量的请求按照我们指定的方式均衡的分配给集群中的每台服务器。
-
保证后端服务安全
因为一般后台服务地址不会暴露,所以使用浏览器不能直接访问,可以把nginx作为请求访问的入口,请求到达nginx后转发到具体的服务中,从而保证后端服务的安全。
3. 谈谈SpringMvc消息转换器的应用
Spring MVC消息转换器是用于在HTTP请求和响应之间进行Java对象和数据格式(如JSON、XML)之间的转换,解决前后端数据交互的问题。
在一个前后端分离的应用中,前端通过Ajax请求发送JSON数据到后端,Spring MVC消息转换器可以将JSON数据转换成Java对象,同时也能将Java对象转换成JSON数据返回给前端。
- Spring MVC消息转换器的默认支持哪些数据格式?
- Spring MVC消息转换器默认支持JSON、XML、表单数据等格式。
- 如何自定义消息转换器?
- 可以通过继承
AbstractHttpMessageConverter
类或实现HttpMessageConverter
接口来自定义消息转换器。然后,在Spring配置中注册这个自定义的消息转换器。
- 可以通过继承
- 你在项目中的哪些场景使用了消息转换器?
- 例如,在RESTful API的开发中,使用JSON格式进行前后端数据的传输。通过配置消息转换器,能够自动将Controller的方法返回的Java对象转换成JSON格式响应给客户端。
4. 你是如何利用AOP面向切面编程?
AOP是什么?解决了什么问题?应用场景?
-
面向切面编程AOP 的目的是将横切关注点(如日志记录、事务管理、权限控制、接口限流、接口幂等等)从核心业务逻辑中分离出来,通过动态代理、字节码操作等技术,实现代码的复用和解耦,提高代码的可维护性和可扩展性。
-
日志记录:自定义日志记录注解,利用 AOP,一行代码即可实现日志记录。
性能统计:利用 AOP 在目标方法的执行前后统计方法的执行时间,方便优化和分析。
事务管理:
@Transactional
注解可以让 Spring 为我们进行事务管理比如回滚异常操作,免去了重复的事务管理逻辑。@Transactional
注解就是基于 AOP 实现的。权限控制:利用 AOP 在目标方法执行前判断用户是否具备所需要的权限,如果具备,就执行目标方法,否则就不执行。例如,SpringSecurity 利用
@PreAuthorize
注解一行代码即可自定义权限校验。接口限流:利用 AOP 在目标方法执行前通过具体的限流算法和实现对请求进行限流处理。
缓存管理:利用 AOP 在目标方法执行前后进行缓存的读取和更新。
AOP应用各个通知类型以及执行时间?
- 分别是前置通知(Before)、后置通知(After)、返回通知(AfterReturning)、异常通知(AfterThrowing)和环绕通知(Around)。
5. redis常用的五种数据类型以及使用场景?
字符串: 适用于缓存、计数器、分布式锁等场景。常常用于存储简单的数据,如用户信息、配置信息、验证码、状态信息等。
哈希:适用于存储对象的属性,特别是对象的字段较多时。例如,存储用户信息、商品信息等。
列表:适用于实现队列、栈,记录操作日志,消息队列等。可以按照插入的顺序存储一系列元素,支持快速的插入和删除操作。
集合: 适用于存储唯一性的元素,例如标签、关注列表等。提供了快速判断某个元素是否存在的能力。
有序集合: 适用于需要根据某个属性排序的情况,例如排行榜、时间线等。可以根据分数范围获取一批元素。
6.微信登录是怎么实现的?
- 小程序端,调用wx.login()获取code,就是授权码。
- 小程序端,调用wx.request()发送请求并携带code,请求开发者服务器(自己编写的后端服务)。
- 开发者服务端,通过HttpClient向微信接口服务发送请求,并携带appId+appsecret+code三个参数。
- 开发者服务端,接收微信接口服务返回的数据,session_key+opendId等。opendId是微信用户的唯一标识。
- 开发者服务端,自定义登录态,生成令牌(token)和openid等数据返回给小程序端,方便后绪请求身份校验。
- 小程序端,收到自定义登录态,存储storage。
- 小程序端,后绪通过wx.request()发起业务请求时,携带token。
- 开发者服务端,收到请求后,通过携带的token,解析当前登录用户的id。
- 开发者服务端,身份校验通过后,继续相关的业务逻辑处理,最终返回业务数据。
7. 如何实现的微信支付功能?
完成微信支付有两个关键的步骤:
1️⃣ 就是需要在商户系统当中调用微信后台的一个下单接口,就是生成预支付交易单。
2️⃣ 就是支付成功之后微信后台会给推送消息。
8.webSocket是什么?有什么应用场景?你的项目中如何应用的?
TCP 协议本身是全双工的,但我们最常用的 HTTP/1.1,虽然是基于 TCP 的协议,但它是半双工的,对于大部分需要服务器主动推送数据到客户端的场景,都不太友好,因此我们需要使用支持全双工的 WebSocket 协议。
在 HTTP/1.1 里,只要客户端不问,服务端就不答。基于这样的特点,对于登录页面这样的简单场景,可以使用定时轮询或者长轮询的方式实现服务器推送(comet)的效果。
对于客户端和服务端之间需要频繁交互的复杂场景,比如网页游戏,都可以考虑使用 WebSocket 协议。
webSocket也有一定的缺点:
- 服务器长期维护长连接需要一定的成本
- 各个浏览器支持程度不一
- WebSocket 是长连接,受网络限制比较大,需要处理好重连
我的项目在来电提醒和催单提醒中用到了。
-
通过WebSocket实现管理端页面和服务端保持长连接状态
-
当客户支付后,调用WebSocket的相关API实现服务端向客户端推送消息
-
客户端浏览器解析服务端推送的消息,判断是来电提醒还是客户催单,进行相应的消息提醒和语言播报
9. 你的数据统计模块是什么意思?怎么做的?
是一些营业额统计、用户统计、订单统计,销量统计等等。利用apache Echarts做的
10. 大流量下如何设计一个全局唯一的订单号?
在大流量的环境下,我们可以通过 redis 的incr
函数实现序列号自增的特性,同时搭配订单的设计规则,从而保证高并发的环境下,订单唯一性,其中的一个订单规则如下:业务编码+年的后 2 位+月+日+秒+订单数,固定长度为16
,这种订单号规则可以保证 100 年不会重复!