面试系列-淘天提前批面试

embedded/2024/10/11 3:19:26/

00-淘天提前批面试

在牛客上看到了淘天提前批的面试题目,这里分析一下淘天面试的问了有哪些内容,面试的重点 是偏向哪些方面

项目相关

1、秒杀架构如何设计?

问了秒杀的架构如何设计,对于秒杀的设计,秒杀符合 写多读少 的场景,那么针对 写比较多 的场景,可以将 Redis 作为 主存储 来操作,这样并发度比较高,Redis 扣减成功之后,再通过 异步 去 DB 中扣减库存,将耗时操作从主干流程中剥离出去,提升主干流程的响应速度,这是从性能的方面进行设计

不过基于 Redis 进行操作的话,Redis 集群部署时,在极端情况下,如果 Redis Master 上扣减库存之后,宕机,此时数据还没来得及同步到 Slave 节点,此时就会出现 库存超卖 的情况,不过这种情况概率很低,并且如果库存超卖的话,只要在订单系统中有防超卖机制即可

2、项目的 QPS 是多少?

这个问的是有没有对项目接口进行压测,一般部署 Prometheus、Grafana 对机器、应用进行监控,再通过 JMeter 进行压测即可,不过说 QPS 时,要先说明机器使用的配置,一般我们学生在学习时,使用 2C4G 服务器进行压测居多(比较便宜)

3、TPS 和 QPS 的区别是什么?

这个就是应用的 性能指标 ,TPS 是指应用每秒处理的事务数量,QPS 是指应用每秒处理的请求数量,一般情况下 QPS 是要多于 TPS 的,因为一个事务中可能包含多个增删改查

基础能力

try catch

问题 :在 try catch finally 中,如果在 try 中 return 了,还会执行 finally 吗?

会执行的,这里应该问的就是 try catch finally 的 return 执行顺序,如下:

1、try、catch 中没有 return 语句

当try和catch中都没有return语句时, 执行顺序依次为:

try --> 执行发生异常 --> catch --> finally

2、try、catch 中有 return 语句,finally 中没有 return。

先把 try 或 catch 中的 return 值保存在局部变量中,再去执行 finally 中的语句,最后 return 返回

3、try、catch 中有 return 语句,且 finally 中也有 return finally 中的 return 最后会覆盖 try、catch 中的 return 数据

问题2 :在 finally 中一般关闭资源,如果出现异常怎么处理?

打印日志进行记录就好了

如果在 finally 中关闭资源要再 try catch 的话,那代码也太臃肿了,我看了 RocketMQ 相关的源码,在 finally 中没有在 finally 中再进行其他的一些重试操作(如果不对,请指正)

final

问题1 : final 放在类上有什么用?

final 放在类上表明该类 不可以被继承 ,主要是为了保证 类的安全性

如果 final 修饰方法,则该方法也不可以被 重写

扩展

  • private 方法会被隐式指定为 final 方法
  • final 修饰的成员变量使用前要初始化

问题2 :final 修饰 HashMap,可以向这个 Map 中添加元素吗?

可以的

final 修饰变量的话,有两种情况:

  • final 修饰基础类型变量 :则数值在初始化之后,就无法更改
  • final 修饰引用类型变量 :则初始化之后,不可以再指向其他对象,但是可以修改该引用类型变量内部的值

多线程

问题1 :五个线程 abcde 如果想先执行 a,再执行 bcd,bcd 执行完后执行 e 如何做?

可以使用 CompletableFuture 来 做多个任务的编排 ,如下:

public class Main {
    public static void main(String[] args) {
        CompletableFuture<Void> taskA = CompletableFuture.runAsync(() -> {
            System.out.println("Thread a is running.");
        });

        CompletableFuture<Void> taskB = taskA.thenRun(() -> {
            System.out.println("Thread b is running.");
        });

        CompletableFuture<Void> taskC = taskA.thenRun(() -> {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            System.out.println("Thread c is running.");
        });

        CompletableFuture<Void> taskD = taskA.thenRun(() -> {
            System.out.println("Thread d is running.");
        });

        CompletableFuture<Void> taskE = CompletableFuture.allOf(taskB, taskC, taskD).thenRun(() -> {
            System.out.println("Thread e is running.");
        });

        taskE.join(); // 等待所有任务执行完成
    }
}

问题2 :用过 CountDownLatch 吗?

CountDownLatch 是 JUC 包下边的工具类,使用场景 为:有多个子任务,此时需要这么多的子任务都执行完毕之后,再去执行主干流程,如果有任何一个任务没有执行完毕都会阻塞等待

如下:

public class Main {
   public static void main(String[] args) throws InterruptedException {

      CountDownLatch countDownLatch = new CountDownLatch(6);

      for (int i = 1; i <= 6; i ++) {
         new Thread(()->{
            System.out.println(Thread.currentThread().getName() + "执行任务");
            countDownLatch.countDown();
         }, String.valueOf(i)).start();
      }

      // 等待子线程执行任务
      countDownLatch.await();

      // 主线程继续向下执行...
   }
}

问题3: Thread.sleep() 有用过吗?会放弃 CPU 锁吗?如果需要让出时间片,该通过什么方式让出?

Thread.sleep() 方法不会让线程放弃 CPU 锁的占用,但是会释放 CPU 时间片

线程还可以通过 Thread.yield() 方法来让出 CPU 时间片,但是可能线程刚刚让出 CPU 时间片,又会重新获取 CPU 时间片继续执行

synchronized

问题1: 你一般会使用 synchronized 锁在方法上吗?如果是锁对象,这个对象如何确定下来呢?

如果是整个方法需要同步的话,那么就可以将 synchronized 放在方法上,synchronized 锁在方法上和锁在对象上的区别就是: 锁的粒度不同

synchronized 锁在对象上,可以让锁的粒度更小,并发度更高

synchronized 锁对象的话,只要保证多个线程去执行 synchronized 并发代码块的时候,可以取到同一个对象即可,也就是保证这个对象是单例的

问题2: synchronized 锁在 this 上可以吗?

可以的,锁定 this 对象就是锁定当前这个实例对象,我们的工程一般都是基于 Spring 的,Spring 中的 Bean 都是单例的,因此通过 synchronized 可以达到同步的效果

HashMap

问题1: HashMap 扩容和树化说一下

HashMap 的 扩容 条件:当 HashMap 中数组的元素个数超过了 当前容量 * 负载因子(默认0.75) 就会触发扩容,会将数组长度扩容为原来的 2 倍

HashMap 的 树化 条件:当数组长度达到 64 并且链表长度超过 8 就会进行树化(如果链表长度超过 8,但是数组长度不到 64,就会对数组扩容 1 次,而不会执行树化)

问题2: 你知道红黑树每次翻转高度差的变化么?

每次翻转之后,要么是左旋要么是右旋,每次旋转之后高度会减 1,来保证稳定

可以在这个网站中自己插入节点看一下红黑树的旋转情况比较直观:cs.usfca.edu

扩展:HashMap 中为什么没有缩容?

因为 HashMap 中 Node 是比较占用空间的,在 remove 时,会将 Node 指针置为 Null,Node 就会被 GC 掉,如果缩容的话,仅仅节省了数组的空间占用,优化并不大

并且缩容操作是要放在 remove 时,那么会导致 remove 的时间复杂度由 O(1) 变为 O(N)

网络

HTTPS

问题1: HTTPS 和 HTTP 的区别?

HTTP 是超文本传输协议,明文传输数据,存在安全风险

HTTPS 使用 HTTP 通信,通过加入 SSL/TLS 安全协议来对数据包进行加密传输

HTTP 是通过 三次握手 来建立连接,而 HTTPS 在 三次握手 之后,还需要进行 SSL/TLS 的握手过程 才可以进入加密报文传输(HTTP默认端口 80、HTTPS 默认端口 443)

问题2: HTTPS 可以防止哪些攻击?

HTTPS 将数据加密传输,可以防止 传输的数据被窃取

扩展: HTTPS 的加密流程

接下来说一下 HTTPS 是如何保证数据不被窃取的:

HTTPS 使用的 对称加密 + 非对称加密 两者结合的算法

HTTPS 在 HTTPS 握手的时候,使用的是 非对称加密 ,服务器会发送给浏览器 数字证书 ,包含了公钥,浏览器使用公钥加密一个随机生成的 对称密钥 ,发送给服务器

当浏览器和服务器建立通信之后,使用对称密钥来进行数据的加密解密,这个过程使用的 对称加密

为什么要使用两种加密算法的结合呢?

  • 对称加密:加密解密过程中使用相同的密钥,速度很快,但是如何让双方都安全的拿到这个密钥比较困难(因此和非对称加密结合,来安全的传输这个对称密钥)
  • 非对称加密:加密解密过程中使用一对密钥,即公钥和私钥。公钥是公开的,用于加密;私钥只能自己拿到,用于解密,整个过程相对复杂,比较耗时,一般用于密钥的交换

通过了解这两种算法的区别,也就知道了为什么要使用这两种算法的结合了,HTTPS 既想要对称加密的性能,又想要非对称加密的安全性!

整个 HTTPS 使用非对称加密以及对称加密的流程如下:

1707122304326
1707122304326

问题3 : HTTPS 建立连接是几次握手?

这里 HTTPS 的相关内容较为复杂,我写的比较简陋,如果我们的项目不是太偏向于网络的话,个人觉得了解整个流程原理即可

HTTPS 的通信流程为:

  • 客户端向服务器请求获取 公钥
  • 双方协商产生 对称密钥
  • 双方采用 对称密钥 进行加密通信

前两个步骤是建立 SSL/TLS 的过程,HTTPS 是基于 SSL 或 TLS 进行加密的,不过 SSL 已经退出历史舞台了,现在说的 HTTPS 其实就是 HTTP+TLS

那么 TLS 握手的过程总共包含了 4 次通信 ,在 4 次通信之后,TLS 协议也就建立成功了,可以进行 HTTPS 通信了,4 次通信如下:

  • 第一次通信 ClientHello:客户端向服务端发送加密请求,主要是协商 TLS 版本、随机数(生成后续的对称密钥)
  • 第二次通信 ServerHello:服务端向客户端回复,主要协商 TLS 版本、随机数(生成后续的对称密钥)、数字证书(包含公钥)
  • 第三次通信 客户端回应:取出数字证书的公钥,将用于通信的 对称密钥 通过公钥加密发送给服务端
  • 第四次通信 服务端最后回应:使用自己本地的密钥进行解密,得到用于通信的 对称密钥

之后双方就可以使用这个 对称密钥 进行加密通信了

问题4: HTTP1 和 HTTP2 的区别?

常用的 HTTP 协议的版本包含了 HTTP/1.1、HTTP/2.0、HTTP/3.0,不过目前常见的就是 HTTP/1.1 和 HTTP/2.0(通过 F12 控制台可以看到协议版本)

  • HTTP1.0 中为 短连接 ,每次通信都要建立一次 TCP 连接,开销很大

  • 因此在 HTTP/1.1 中优化为 长连接 ,通过管道机制在一个 TCP 连接中,客户端可以发送多个请求,服务端可能会按顺序处理请求,因此会导致 后续的请求被阻塞

  • HTTP/2 是基于 HTTPS 的,相对于 HTTP/1 来说 比较安全 ,并且 HTTP/2 提出了 Stream 的概念,可以在一个 TCP 连接上维护多个 Stream,通过多个 Stream 可以并行发送消息,并且 HTTP/2 将请求和响应数据分割为了更小的帧 ,减少传输无用数据的体积

HTTP/2 主要就在这 3 个方面做出了优化:安全性、Stream、帧

场景题

最后一个场景题:手机淘宝输入一个商品按下搜索之后,会发生一个怎么样的过程?

这个题如果大家有比较好的想法,可以讨论一下!

本文由 mdnice 多平台发布


http://www.ppmy.cn/embedded/125683.html

相关文章

信息学奥赛一本通 1885:【14NOIP提高组】寻找道路 | 洛谷 P2296 [NOIP2014 提高组] 寻找道路

【题目链接】 洛谷 P2296 [NOIP2014 提高组] 寻找道路 ybt 1885&#xff1a;【14NOIP提高组】寻找道路 【题目考点】 1. 图论&#xff1a;广搜 2. 图论&#xff1a;反图 【解题思路】 设path数组&#xff0c;path[i]表示顶点i出发到终点t是否有路径。 先求path数组&#…

python 实现markov chain马尔可夫链算法

markov chain马尔可夫链算法介绍 马尔可夫链&#xff08;Markov Chain&#xff09;算法是一种基于概率的随机过程模型&#xff0c;用于描述状态之间的转移规律。以下是关于马尔可夫链算法的一些详细介绍&#xff1a; 定义与特性 定义&#xff1a;马尔可夫链是指具有马尔可夫…

大数据新视界 --大数据大厂之 从 Druid 和 Kafka 到 Polars:大数据处理工具的传承与创新

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

【重学 MySQL】五十、添加数据

【重学 MySQL】五十、添加数据 使用INSERT INTO语句添加数据基本语法示例插入多行数据注意事项 使用LOAD DATA INFILE语句批量添加数据其他插入数据的方式注意事项 在MySQL中&#xff0c;添加数据是数据库操作中的基本操作之一。 使用INSERT INTO语句添加数据 使用 INSERT IN…

大数据新视界 --大数据大厂之 Druid 查询性能提升:加速大数据实时分析的深度探索

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

论文阅读:InternVL v1.5| How Far Are We to GPT-4V? 通过开源模型缩小与商业多模式模型的差距

论文地址&#xff1a;https://arxiv.org/abs/2404.16821 Demo&#xff1a; https://internvl.opengvlab.com Model&#xff1a;https://huggingface.co/OpenGVLab/InternVL-Chat-V1-5 公开时间&#xff1a;2024年4月29日 InternVL1.5&#xff0c;是一个开源的多模态大型语言模…

facebook受众选择设置策略的最佳方式

在进行Facebookguanggao投放时&#xff0c;受众的选择是一个至关重要的步骤。正确的受众选择不仅能够帮助我们更好地定位目标用户&#xff0c;还能显著提高guanggao的转化率和投资回报率&#xff08;ROI&#xff09;。然而&#xff0c;受众选择的数量和范围同样是需要认真考虑的…

spring boot发送邮件

文章目录 项目目录结构添加maven依赖application.yml配置发信人信息编码测试创建 Email 工具类 EmailUtil测试发送邮件 项目目录结构 添加maven依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-mail&l…