Java100道面试题

server/2025/1/12 12:20:26/

1.JVM内存结构

1. 方法区(Method Area)

方法区是JVM内存结构的一部分,用于存放类的相关信息,包括:

  • 类的结构(字段、方法、常量池等)。
  • 字段和方法的描述,如名称、类型、访问修饰符等。
  • 静态变量。
  • Java虚拟机在运行时加载的类的信息。

方法区通常在堆区的外部,与堆区是分开的。因为主要存放的是类的信息和静态数据,所以这一部分数据的生命周期与类本身相同。

2. 堆(Heap)

堆是JVM中用于动态分配内存的区域,是对象和数组的存储区。堆内存是JVM运行过程中最大的内存区域,所有对象实例和数组都是在这个区域中分配的。堆区又可以分为几个部分:

  • 新生代(Young Generation):用于存放新创建的对象,年轻代又可以分为三个部分:Eden空间和两个Survivor空间。新创建的对象首先在Eden区分配,经过垃圾回收后存活下来的对象会被移动到Survivor区。
  • 老年代(Old Generation):用于存放长期存活的对象。经过多次垃圾回收,仍然存活的对象会被转移到老年代。
  • 持久代(PermGen)/元空间(Metaspace):在旧版本的JVM中,方法区使用持久代来存储类元信息。在Java 8及其之后的版本中,持久代被元空间替代,主要用于存储类的元数据,不再属于Java虚拟机的堆内存,而是使用本地内存。
3. 虚拟机栈(VM Stack)

虚拟机栈用于存储局部变量、操作数栈、动态链接等信息。每个线程都有自己的虚拟机栈。栈是先进后出(LIFO)的结构,栈中的每一个栈帧(Stack Frame)对应一个方法的调用。在方法调用时,相关的局部变量、参数值以及方法返回地址等都会被推入栈中,而在方法返回时,栈帧会被弹出。

4. 本地方法栈(Native Method Stack)

本地方法栈用于支持JVM调用本地(Native)方法,类似于虚拟机栈。它存储的是本地方法的栈帧。不同的JVM实现可能会有所不同,但其功能主要是协助执行 Java 与其他语言(如C、C++)交互的本地方法。

5. 程序计数器(Program Counter Register)

程序计数器是一个较小的内存区域,保存着当前线程所执行的字节码的地址。它是线程私有的,每个线程都有一个程序计数器,以便在多线程环境中跟踪每个线程的执行点。可以理解为一个指针,指向当前正在执行的指令。

2.垃圾回收机制(GC)

垃圾回收算法

垃圾回收算法有多种,常见的有:

  • 标记-清除(Mark-and-Sweep)

    • 该算法分为两个阶段。第一阶段“标记”会遍历所有可达对象并标记它们,第二阶段“清除”会删除没有被标记的对象。尽管这个方法简单有效,但它可能导致内存碎片。
  • 复制(Copying)

    • 该算法将内存分为两个空间(通常称为“From”和“To”区域),在进行垃圾回收时复制所有活跃对象到另一个区域。复制后,原来的区域会被清空。该方法避免了内存碎片,但需要更多内存。
  • 标记-整理(Mark-and-Compact)

    • 该算法与标记-清除结合。首先,它会标记所有可达对象,然后将它们移动到一端,以消除内存碎片,并清理未使用的内存。
  • 分代收集(Generational Collection)

    • 根据对象的生命周期将内存分为若干代(通常为年轻代和老年代)。新创建的对象通常会在年轻代中变化较快,因此通过频繁回收年轻代来提高效率。而老年代中的对象较少变化,因此不需要频繁检查和回收。

JVM的垃圾回收机制:GC,是Java提供的对于内存自动回收的机制。

GC(Garbage Collection)是Java虚拟机(JVM)中的一项重要功能,用于自动管理堆内存中不再使用的对象,释放其占用的内存空间。GC通过标记和回收无效对象来实现内存的回收和释放,以避免内存泄漏和溢出。

        下面是GC的主要工作原理和过程:

        1、对象的标记:GC首先标记出所有活动对象,即仍然被引用或可达的对象。它从一组根对象开始,逐步遍历对象图,将可达的对象标记为活动对象,未标记的对象则被认为是无效的。

        2、垃圾回收:在标记完成后,GC会对未标记的对象进行回收。具体的回收算法可以是不同的,常见的算法包括标记-复制算法、三色标记算法等。

        3、内存压缩和整理:某些垃圾回收算法在回收完对象后,可能会产生内存碎片。为了优化内存使用,GC可能会进行内存压缩和整理操作,使得分配的对象在内存中连续存放,减少内存碎片的影响。

        GC的优点是可以自动管理内存,减少了手动内存管理的复杂性,避免了内存泄漏和溢出的问题。但是,GC也会带来一定的性能开销。因此,在开发Java应用程序时,需要合理配置GC的参数和调整垃圾回收策略,以平衡性能和内存的使用。
 

3.JVM调优的方法?

1. 内存管理调优
  • 堆内存大小设置:可以通过 (初始堆大小)和 (最大堆大小)参数来设置堆内存的大小。例如:。根据应用程序的需求来调整堆的大小

  • 选择合适的垃圾回收器:不同的垃圾回收器适用于不同的场景,例如:

    • G1垃圾回收器():适合大堆内存和低延迟的应用。
    • Parallel垃圾回收器():适合高吞吐量的应用。
    • CMS垃圾回收器():适合需要最小停顿的应用。
    • ZGC和Shenandoah:适合需要极低延迟的应用,但需要JVM版本支持。
  • 调整新生代和老年代的比例:可以通过 或 参数来调整新生代与老年代的比例,以优化对象的存活率和垃圾回收频率。

2. 垃圾回收调优
  • 影响GC频率和停顿时间的参数

    • MaxGCPauseMillis:设置最大GC停顿时间。GC会尝试满足这个时间限制。
    • GCTimeRatio:调整应用可用时间与GC时间的比例。
  • 监控和分析:使用JVM提供的工具(如JVisualVM、JConsole、GC日志)来监控和分析垃圾回收的情况,识别GC问题。

3. JIT编译优化
  • 开启Tiered Compilation:使用,可以提高启动性能。

  • 调整JIT编译行为:通过设置编译阈值,以控制何时将方法编译为机器代码。-

4. 线程调优
  • 线程栈大小:可以用参数设置每个线程的栈大小。例如:。适当调整以减少栈溢出和内存使用。

  • 线程池管理:如果应用使用线程池,确保合理配置线程数,避免因过多线程上下文切换引起的性能问题。

5. 其他优化选项
  • 类加载与内存优化

    • 使用 可保持对象引用的压缩,有助于减少堆内存的使用。
  • 禁用或调整assertions:在生产环境中,可以禁用不必要的断言以提高性能:。-da

6. 性能监控与分析
  • 使用JVM监控工具:如Java Mission Control(JMC)和Flight Recorder来分析应用性能,查找瓶颈。

  • Profiling:使用工具(如YourKit、VisualVM等)进行代码剖析,找到性能开销较大的热点代码。

7. 配置文件与启动参数
  • 合理配置启动参数:根据具体的应用场景和服务器配置调整JVM的启动参数,比如开启JIT、选择合适的GC、调整堆内存大小等。

4.堆和栈的区别?

1. 结构和用途:
  • 栈(Stack)

    • 栈是一种先进后出(LIFO, Last In First Out)的数据结构。
    • 用于存储局部变量、函数参数和函数调用信息等。在函数调用时,函数的局部变量会压入栈中,函数返回时会将这些变量从栈中弹出。
    • 每个线程都有自己的栈,用于跟踪函数调用和局部变量。
  • 堆(Heap)

    • 堆是一种基于动态存储分配的内存区域,没有特定的结构(可以认为是自由存储区)。
    • 用于存储动态分配的对象和数据,如使用 关键字在C++中分配的对象,或在Java中创建的对象。
    • 堆内存可以由程序员直接管理,程序员需要手动申请和释放内存,或者由垃圾回收机制(如Java、C#)自动管理。
2. 存储管理:
    • 存储在栈内存中的数据管理简单,入栈和出栈操作的时间复杂度都是O(1)。
    • 由于栈的大小是固定的(通常由操作系统限定),栈溢出(Stack Overflow)可能导致程序崩溃。
    • 堆内存的大小通常只受到系统内存的限制,算法的复杂性和速度依赖于内存分配和回收的实现。
    • 堆内存分配需要更多的时间,且容易出现内存泄漏或内存碎片问题。
3. 生命周期:
    • 数据的生命周期由函数调用决定。函数结束后,栈上的数据会自动释放。
    • 数据的生命周期由程序员控制,只有在不再需要对象时,才能显式地释放空间。如果没有释放,则会造成内存泄漏。
4. 访问速度:
    • 由于其结构简单,内存访问速度较快,通常会比堆更快。
    • 由于堆内存的管理要复杂得多,非顺序访问可能导致较慢的访问速度。

5.JVM使用命令

jvm 的一些命令_jvm命令-CSDN博客

jinfo 

描述:输出给定 java 进程所有的配置信息。包括 java 系统属性和 jvm 命令行标记等。

jstat

jstat -gcutil <pid> <interval>

查看java进程的gc情况

以百分比显示每个区域的内存使用情况;

参数interval表示每多少毫秒刷新一次

6.class.forName和Classload的区别

1.相同点
两者都可以对类进行加载。
对于任意一个类/对象,我们都能够通过反射能够调用这个类的所有属性方法。

2.不同点
抽象类ClassLoader中实现的方法loadClass,loadClass只是加载,不会解析更不会初始化所反射的类。常用于做懒加载,提高加载速度,使用的时候再通过.newInstance()真正去初始化类。 

7.谈谈JDK1.8特性有哪些

Lambda表达式 类似于ES6中的箭头函数
接口的默认方法和静态方法
新增方法引用格式
新增Stream类
新的日期API,Datetime,更方便对日期的操作
引入Optional,在SpringData中使用较多,然后再通过get获取值,主要用于防止NPE。
支持Base64
注解相关的改变
支持并行(parallel)数组
对并发类(Concurrency)的扩展。
JavaFX。

JDK1.8常用新特性_jdk1.8的新特性-CSDN博客

8.反射获取类中的所有方法和获取类中的所有属性

  • getDeclaredMethods() 和 getDeclaredFields() 允许你访问类的所有方法和字段,包括私有的。
  • getMethods() 和 getFields() 只返回公共方法和字段,且包括继承的父类方法和字段。

9.懒汉和饿汉模式的区别?口述两种模式。 

在饿汉式单例模式中,“饿” 体现的是一种急切的状态。就好像一个很饿的人,在看到食物(这里类比于单例对象)的时候,会迫不及待地先把食物拿到手(创建单例对象)。在这个模式下,单例对象在类加载阶段就被创建出来,而不是等到真正需要使用这个对象的时候才去创建。这种方式比较急切,所以被称为 “饿汉模式”。

在懒汉模式下,实例在第一次使用时才进行创建,因此称为“懒汉”,在需要被用的时候被创建,突出一个字“懒”

10.如何保证单例模式在多线程中的线程安全

私有构造函数:通过将构造函数设置为私有,我们禁止了外部类直接实例化Singleton类。这确保了只能通过getInstance()方法来获取实例。

volatile关键字:volatile关键字确保了instance变量的可见性。当一个线程修改了一个volatile变量的值时,其他线程能够立即看到这个变化。这样可以避免线程之间的缓存不一致问题。

双重检查锁定(Double-Checked Locking):为了减少同步开销,我们在第一次检查instance是否为空后,才进入同步块。这是因为大多数情况下,实例已经被创建,不需要进入同步块。只有在第一次创建实例时才会需要同步。

同步块:在同步块内部,我们再次检查instance是否为空,以确保只有一个线程能够创建实例。这是为了防止多个线程同时通过了第一个空检查并尝试创建多个实例。

11.spring运用了什么设计模式?讲一下哪些部分运用到了这些设计模式

  • 一、简单工厂模式(Simple Factory)
  • 定义:

    简单工厂模式:并不属于 GoF(四人组)总结的 23 种设计模式,但它却在实际开发中被频繁使用。其核心是 由一个工厂类根据传入的参数,动态决定创建哪一个产品类的实例。

    举例:

    Spring 中的 BeanFactory 就是简单工厂模式的体现,根据传入一个唯一的标识来获得 Bean 对象。但是,在传入参数后创建 Bean 还是传入参数前创建 Bean,这个要根据具体情况而定。

  • 二、工厂方法模式(Factory Method)
  • 定义:

    工厂方法模式:定义了一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到子类。

    举例:

    Spring 使用工厂模式可以通过 BeanFactory 或 ApplicationContext 创建 Bean 对象。两者对比如下:

    BeanFactory:延迟注入(使用到某个 Bean 的时候才会注入),相比于 ApplicationContext 来说会占用更少的内存,程序启动速度更快。
    ApplicationContext:容器启动的时候,不管你用没用到,一次性创建所有 Bean。BeanFactory 仅提供了最基本的依赖注入支持,ApplicationContext 扩展了 BeanFactory,除了有 BeanFactory 的功能还有额外更多功能,所以一般开发人员使用 AplicationContext 更多。

  • 三、单例模式(Singleton)
  • 定义:

    单例模式:确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。

    举例:

    在我们系统中,有一些对象其实我们只需要一个,比如说:线程池、缓存、对话框、注册表、日志对象、充当打印机、显卡等设备驱动程序的对象。事实上,这一类对象只能有一个实例,如果制造出多个实例就可能会导致一些问题的产生,比如;程序的行为异常、资源使用过量、或者不一致性的结果。

  • 四、适配器模式(Adapter)
  • 定义:

    适配器模式:将一个接口转换成客户希望的另一个接口,适配器使接口不兼容的那些类可以一起工作。

    举例:

    Spring AOP 中的适配器模式:

    我们知道 Spring AOP 的实现是基于代理模式,但是 Spring AOP 的增强或通知(Advice)使用到了适配器模式,与之相关的接口是 AdvicorAdapter。

  • 五、代理模式(Proxy)
  • 定义:

    代理模式:为其他对象提供一个代理以控制对这个对象的访问。

    举例:

    代理模式在 AOP 中的应用:

    AOP(Aspect-Oriented Programming,面向切面编程):能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任(例如:事务处理、日志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可扩展性和可维护性

  • 七、观察者模式(Observer)
  • 定义:

    观察者模式:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖它的对象都得到通知并被自动更新。

    举例:

    Spring 事件驱动模型就是观察者模式很经典的一个应用。Spring 事件驱动模型非常有用,在很多场景都可以解耦我们的代码。比如我们每次添加商品的时候都需要重新更新商品索引,这个时候就可以利用观察者模式来解决这个问题

  • 八、策略模式(Strategy)
  • 定义:

    策略模式:定义了一系列的算法,并将每一个算法封装起来,使它们可以互相替换。策略模式让算法独立于使用它的客户。

    举例:

    Spring 框架的资源访问 Resource 接口。该接口提供了更强的资源访问能力,Spring 框架本身大量使用了 Resource 接口来访问底层资源。

  • 九、模板方法模式(Template Method)
  • 定义:

    模板方法模式:在一个方法中定义了一个算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法的某些步骤。

    举例:

    Spring 中 JdbcTemplate、HibernateTemplate 等以 Template 结尾的对接入内容进行操作的类,它们就使用到了模板方法。一般情况下,我们都是使用继承的方法来实现模板模式,但是 Spring 并没有使用这种方式,而是使用 Callback 模板与模板方法模式配合,既达到了代码复用的效果,同时增加了灵活性。

12.说说你都知道哪些设计模式?最常用的有哪些?总共有几种设计模式?

23种设计模式 - 知乎

13.redis可以储存哪几种数据类型?你的项目里都用redis存储哪些数据

  • 字符串类型(String)‌:这是Redis中最基础的数据类型,可以存储任何形式的字符串,包括文本、序列化后的对象、二进制数据等。字符串类型的Value最多可以容纳512MB的数据‌。

  • 哈希类型(Hash)‌:哈希类型可以看作是一个键值对的集合,适合存储对象的信息。例如,一个用户的信息可以存储为哈希类型,包含姓名、密码、年龄等字段‌。

  • 列表类型(List)‌:列表类型是一个简单的字符串列表,可以存储一系列的字符串元素。它支持按照插入顺序排序的字符串列表,常用于实现队列和栈等数据结构‌。

  • 集合类型(Set)‌:集合类型是一个无序的字符串集合,支持添加、删除和判断元素是否存在等操作。集合中的元素是唯一的,常用于实现交集、并集和差集等操作‌。

  • 有序集合类型(Zset)‌:有序集合类型是一个集合,其中的每个成员都关联一个双精度浮点数分数,这使得它既可以像集合一样存储不重复的元素,又可以像排序的列表一样返回有序的数据。常用于排行榜等场景‌

  • 字符串类型‌:常用于缓存数据库查询结果、实现计数器功能(如网站访问量计数)等‌。
  • 哈希类型‌:适合存储对象信息,如用户信息、商品信息等‌。
  • 列表类型‌:常用于实现消息队列、任务队列等功能‌。
  • 集合类型‌:适用于需要去重的数据存储,如好友关系、标签等‌。
  • 有序集合类型‌:适用于需要排序的数据存储,如排行榜、评分系统等‌

14.redis有哪些常用操作?(命令)

键操作
  1. SET key value
    设置键 key 的值为 value

  2. GET key
    获取键 key 的值。

  3. DEL key
    删除键 key

  4. EXISTS key
    判断键 key 是否存在,存在返回 1,不存在返回 0。

  5. EXPIRE key seconds
    设置键 key 的过期时间为 seconds 秒。

  6. TTL key
    获取键 key 的剩余过期时间(单位秒)。

  7. RENAMENX oldkey newkey
    重命名键 oldkeynewkey,如果 newkey 已存在,重命名失败。

  8. TYPE key
    获取键 key 的数据类型。

  9. FLUSHDB
    删除当前数据库的所有键。

  10. FLUSHALL
    删除所有数据库的所有键

Pub/Sub(发布/订阅)

Redis 支持发布/订阅模型,用于实时消息传递。

  1. PUBLISH channel message
    向频道 channel 发布消息 message

  2. SUBSCRIBE channel [channel ...]
    订阅一个或多个频道。

  3. UNSUBSCRIBE [channel ...]
    取消订阅一个或多个频道。

  4. PSUBSCRIBE pattern [pattern ...]
    订阅一个或多个模式匹配的频道。

  5. PUNSUBSCRIBE [pattern ...]
    取消订阅模式匹配的频道

字符串操作
哈希操作
列表操作
集合操作
有序集合操作
事务操作
其他常用命令

15.redis缓存和数据库怎么保持一致?

1. 读写策略调整

先更新数据库,再更新缓存

原理:这种策略在更新数据时,先对数据库执行更新操作,然后立即更新缓存中的数据。这样可以尽量保证缓存和数据库的数据同步。

缺点:存在一定的风险。如果更新缓存失败,就会导致数据库和缓存数据不一致。而且,在高并发场景下,可能会出现多个线程同时更新缓存,导致缓存数据的更新顺序混乱,进而引发数据不一致的情况。

先删除缓存,再更新数据库

原理:当需要更新数据时,先将缓存中的对应数据删除。这样,下一次读取数据时,由于缓存中没有数据,就会从数据库中读取最新的数据,并将其放入缓存。

缺点:在高并发场景下可能会出现问题。例如,一个线程 A 删除了缓存,但是还没来得及更新数据库,此时另一个线程 B 来读取数据,发现缓存为空,就会从数据库读取旧数据并放入缓存。然后线程 A 再更新数据库,这样就导致了缓存和数据库的数据不一致。

2. 使用缓存更新队列

原理:将缓存更新操作放入一个队列中,由专门的消费者线程来处理这些更新操作。当有数据需要更新时,先将更新消息发送到队列中,这样可以避免多个线程同时更新缓存导致的混乱。

优点:可以实现异步更新缓存,减少对数据库更新操作的直接依赖,提高系统的并发性能。并且能够保证缓存更新的顺序,降低数据不一致的风险。

缺点:增加了系统的复杂性,需要维护一个消息队列和相应的消费者线程。而且如果队列处理不及时,可能会导致缓存更新延迟。

3. 设置合理的缓存过期时间

原理:为缓存中的数据设置过期时间,当缓存过期后,下一次读取数据时就会从数据库中重新获取最新的数据,从而更新缓存。通过合理设置过期时间,可以在一定程度上减少数据不一致的情况。

举例:对于一些数据更新频率较低的业务场景,如电商系统中的商品分类信息,可以设置较长的缓存过期时间,比如一天。而对于更新频率较高的数据,如商品的库存信息,可以设置较短的缓存过期时间,如几分钟。

缺点:如果过期时间设置过长,可能会导致数据不一致的时间过长;如果过期时间设置过短,会频繁地从数据库读取数据,增加数据库的压力。

4. 采用数据库的事务机制与缓存操作结合

原理:在更新数据库的操作中加入事务机制,确保数据库更新成功后再进行缓存操作。如果缓存操作失败,可以通过事务回滚来保证数据的一致性。

示例:在关系型数据库(如 MySQL)中,可以使用存储过程来包裹数据库更新和缓存更新操作。在存储过程中,开启一个事务,先更新数据库表,然后更新缓存。如果缓存更新失败,就回滚事务,从而保证数据库和缓存数据的一致性。

缺点:这种方法会增加数据库的负担,并且需要对业务逻辑进行比较复杂的改造,以适应事务和缓存操作的结合。

5. 读取时对比缓存和数据库数据

原理:在读取数据时,同时获取缓存中的数据和数据库中的数据,然后进行对比。如果发现数据不一致,就以数据库数据为准更新缓存。

缺点:这种方法会增加读取数据的时间成本,因为每次读取都需要从数据库获取数据进行对比。而且在高并发情况下,频繁地对比和更新缓存可能会导致性能下降。

16.redis可以持久化么?如何持久化?

Redis的三种持久化方法详解_redis appendfsync-CSDN博客

17.redis分布式锁怎么使用?

Redis实现分布式锁的7种方案,及正确使用姿势!_redis锁使用-CSDN博客

18.redis缓存数据丢失怎么使用?

Redis如何解决数据丢失问题_redis 数据丢失-CSDN博客

19.redis如果崩溃了如何快速恢复?

Redis宕机了,如何恢复数据_redis宕机数据如何恢复-CSDN博客

20.redis是如何部署的?是单个部署还是集群部署?为什么这么做?

单机模式

新手入门模式。单机模式意味着 Redis 是单点的,部署在一台服务器,挂了就挂了,用在本地测试还可以,但是生产环境就算了。

优势
  1. 部署简单
  2. 省钱,一台服务器就可以了
  3. 不涉及主从复制等,数据强一致
劣势
  1. 单点意味着稳定性基本上为 0,挂了就挂了
  2. 吞吐量受限于单机资源

主从模式

当流量越来越大,单台机器资源不能无限增长,就需要水平扩展到多个节点,使用多个节点分散承接读流量。

主从模式为主节点承接写流量,从节点承接读流量,二者数据一致通过主节点异步复制(全量复制 / 增量复制)到从节点。

优势
  1. 读流量被分摊到多个节点上,读流量支持力度变大
  2. 当主节点宕机/不可用时,可以手动切换主节点继续提供服务
劣势
  1. 当主节点宕机/不可用时手动切换节点,切换过程中 redis (主节点)不可用,并且会丢失主节点 / 从节点之间未同步的数据
  2. 稳定性还是不够,依赖手动切换。不适用于生产。
  3. 写流量还是让主节点独自承受,写流量还是靠单机资源支撑

哨兵模式

哨兵模式主要解决主从模式中手动切换的部分,本质上哨兵代替了人,通过 gossip 协议监控主节点的健康情况。

优势
  1. 不用手动切换主节点了,切换过程中虽然 redis 也是不可用的,但是这个时间会极大的降低
劣势
  1. sentinel 与主节点多了一层心跳检测,有可能 sentinel 与主节点的网络抖动导致重新选举主节点。
  2. redis 主从节点吞吐因心跳检测可能稍微降低。

集群模式

集群模式主要解决了两个问题:写流量水平扩展 & 哨兵与主节点的网络抖动。

集群模式主要的架构为:主节点平分 16384 个槽,集群支持主节点的动态上线/下线(需要 rehash),主节点与从节点通过心跳关联,主节点失联后从节点有权发起选举成为主节点(raft 算法)。

优势
  1. 自管理集群内主从节点上下线,减少因外部集群网络抖动之类的发起的无效选举
  2. 数据按照 slot 存放在多个节点,客户端通过服务端主节点的重定向跳转到具体的槽,可动态调整数据分布
  3. 减少了集群整体不可用的概率,某一主节点宕机只影响一部分数据的访问
  4. 写流量 & 数据平分到多个节点,集群的写请求瓶颈得到缓解
劣势
  1. 集群间状态同步使用 gossip 协议,节点数较多存在较多的心跳网络流量
  2. 主节点的上线/下线需要进行 rehash ,当节点内数据较多耗时较长

redis 节点间复制有两种:全量复制 & 部分复制

全量复制

出现场景
  1. 从节点刚上线需要同步主节点的数据
  2. 从节点切换脑裂后从节点偏移量与主节点不一致的时间点
  3. 从节点偏移量不在主节点的复制缓冲区中
过程
  1. 从节点向主节点发起同步数据的请求
  2. 主节点通过 bgsave 形成当前数据的快照,发给从节点
  3. 从节点删除历史数据,加载主节点发过来 RDB 文件
  4. 从节点拉取主节点缓冲区数据,加载到自身的内存中,并更新当前的偏移量

部分复制

出现场景
  1. 全量复制出现场景之外的场景
  2. 主从日常复制

过程

  1. 主节点将命令同步到缓冲区(AOF)
  2. 从节点拉取缓冲区数据,更新到自身的节点中,并更新当前的偏移量

21.什么是缓存穿透、缓存击穿、缓存雪崩?如何解决?

一、缓存穿透
1、缓存穿透理解 缓存穿透是指查询一个根本不存在的数据,缓存层和持久层都不会命中。在日常工作中出于容错的考虑,如果从持久层查不到数据则不写入缓存层,缓存穿透将导致不存在的数据每次请求都要到持久层去查询,失去了缓存保护后端持久的意义。

缓存穿透示意图:

缓存穿透问题可能会使后端存储负载加大,由于很多后端持久层不具备高并发性,甚至可能造成后端存储宕机。通常可以在程序中统计总调用数、缓存层命中数、如果同一个Key的缓存命中率很低,可能就是出现了缓存穿透问题。 造成缓存穿透的基本原因有两个。第一,自身业务代码或者数据出现问题(例如:set 和 get 的key不一致),第二,一些恶意攻击、爬虫等造成大量空命中(爬取线上商城商品数据,超大循环递增商品的ID)

2、解决方案

2.1 缓存空对象

缓存空对象:是指在持久层没有命中的情况下,对key进行set (key,null)

缓存空对象会有两个问题:

第一,value为null 不代表不占用内存空间,空值做了缓存,意味着缓存层中存了更多的键,需要更多的内存空间,比较有效的方法是针对这类数据设置一个较短的过期时间,让其自动剔除。

第二,缓存层和存储层的数据会有一段时间窗口的不一致,可能会对业务有一定影响。例如过期时间设置为5分钟,如果此时存储层添加了这个数据,那此段时间就会出现缓存层和存储层数据的不一致,此时可以利用消息系统或者其他方式清除掉缓存层中的空对象。

2.2 布隆过滤器拦截

在访问缓存层和存储层之前,将存在的key用布隆过滤器提前保存起来,做第一层拦截,当收到一个对key请求时先用布隆过滤器验证是key否存在,如果存在在进入缓存层、存储层。可以使用bitmap做布隆过滤器。这种方法适用于数据命中不高、数据相对固定、实时性低的应用场景,代码维护较为复杂,但是缓存空间占用少。

布隆过滤器实际上是一个很长的二进制向量和一系列随机映射函数。布隆过滤器可以用于检索一个元素是否在一个集合中。

它的优点是空间效率和查询时间都远远超过一般的算法,缺点是有一定的误识别率和删除困难。

布隆过滤器拦截的算法描述:

初始状态时,BloomFilter是一个长度为m的位数组,每一位都置为0。

添加元素x时,x使用k个hash函数得到k个hash值,对m取余,对应的bit位设置为1。

判断y是否属于这个集合,对y使用k个哈希函数得到k个哈希值,对m取余,所有对应的位置都是1,则认为y属于该集合(哈希冲突,可能存在误判),否则就认为y不属于该集合。可以通过增加哈希函数和增加二进制位数组的长度来降低错报率。

3、两种方案的对比

二:缓存击穿
1、缓存击穿的理解

系统中存在以下两个问题时需要引起注意:

当前key是一个热点key(例如一个秒杀活动),并发量非常大。 重建缓存不能在短时间完成,可能是一个复杂计算,例如复杂的SQL、多次IO、多个依赖等。 在缓存失效的瞬间,有大量线程来重建缓存,造成后端负载加大,甚至可能会让应用崩溃。

2、解决方案

2.1 分布式互斥锁 只允许一个线程重建缓存,其他线程等待重建缓存的线程执行完,重新从缓存获取数据即可。set(key,value,timeout)

2. 永不过期 从缓存层面来看,确实没有设置过期时间,所以不会出现热点key过期后产生的问题,也就是“物理”不过期。 从功能层面来看,为每个value设置一个逻辑过期时间,当发现超过逻辑过期时间后,会使用单独的线程去更新缓

3、两种方案对比:

分布式互斥锁:这种方案思路比较简单,但是存在一定的隐患,如果在查询数据库 + 和 重建缓存(key失效后进行了大量的计算)时间过长,也可能会存在死锁和线程池阻塞的风险,高并发情景下吞吐量会大大降低!但是这种方法能够较好地降低后端存储负载,并在一致性上做得比较好。

永远不过期:这种方案由于没有设置真正的过期时间,实际上已经不存在热点key产生的一系列危害,但是会存在数据不一致的情况,同时代码复杂度会增大。

三:缓存雪崩 1、概念理解

如果缓存集中在一段时间内失效,发生大量的缓存穿透,所有的查询都落在数据库上,造成了缓存雪崩。

这个没有完美解决办法,但可以分析用户行为,尽量让失效时间点均匀分布。大多数系统设计者考虑用加锁或者队列的方式保证缓存的单线程(进程)写,从而避免失效时大量的并发请求落到底层存储系统上。

2、解决方案

2.1 缓存层高可用: 可以把缓存层设计成高可用的,即使个别节点、个别机器、甚至是机房宕掉,依然可以提供服务。利用sentinel或cluster实现。 2.2 做二级缓存,或者双缓存策略: 采用多级缓存,本地进程作为一级缓存,redis作为二级缓存,不同级别的缓存设置的超时时间不同,即使某级缓存过期了,也有其他级别缓存兜底 2.3 数据预热: 可以通过缓存reload机制,预先去更新缓存,再即将发生大并发访问前手动触发加载缓存不同的key,设置不同的过期时间,让缓存失效的时间点尽量均匀 2.4 加锁排队. 限流– 限流算法. 1.计数 2.滑动窗口 3. 令牌桶Token Bucket 4.漏桶 leaky bucket [1] 在缓存失效后,通过加锁或者队列来控制读数据库写缓存的线程数量。比如对某个key只允许一个线程查询数据和写缓存,其他线程等待。 业界比较常用的做法,是使用mutex。简单地来说,就是在缓存失效的时候(判断拿出来的值为空),不是立即去load db,而是先使用缓存工具的某些带成功操作返回值的操作(比如Redis的SETNX或者Memcache的ADD)去set一个mutex key,当操作返回成功时,再进行load db的操作并回设缓存;否则,就重试整个get缓存的方法。 SETNX,是「SET if Not eXists」的缩写,也就是只有不存在的时候才设置,可以利用它来实现锁的效果。

22.List和set和map的区别?

1. 有序性
List:有序,维护元素的插入顺序,可以通过索引访问。
Set:无序(除 LinkedHashSet 保持插入顺序外),不允许重复元素。
Map:键无序(LinkedHashMap 保持插入顺序,TreeMap 按键排序),值无序。
2. 重复性
List:允许存储重复的元素。
Set:不允许存储重复的元素。
Map:键不允许重复,值可以重复。
3. 访问方式
List:按索引访问,通过 get(index) 方法。
Set:只能通过迭代器遍历,不能按索引访问。
Map:通过键访问相应的值,使用 get(key) 方法。
4. 存储结构
List:单一元素的集合,每个元素都有其位置(索引)。
Set:单一元素的集合,但保证元素的唯一性。
Map:键值对的集合,通过键访问对应的值。
五、常见使用场景举例
List:适用于需要按顺序存储和访问元素的场景,如学生名单、任务队列、订单列表等。

Set:适用于需要存储唯一元素的场景,如用户名列表、唯一标识符集合、无重复的集合运算等。

Map:适用于需要通过键快速查找对应值的场景,如字典、配置参数、映射表等。

23.arrarylist,linkedlist;arraylist内部扩容机制是怎样的?

ArrayList和LinkedList_linkedlist扩容机制-CSDN博客

24.hashmap扩容机制,HashMap的底层原理

(转载) HashMap底层原理 + 扩容机制_hashmap的扩容机制-CSDN博客

25.hashmap的底层用什么储存的?线程是否安全?

HashMap的底层数据结构是数组+链表+红黑树‌。HashMap使用哈希表来实现键值对的存储,当发生哈希冲突时,会使用链表来存储冲突的键值对。当链表的长度大于8时,会将链表转换为红黑树,以提高查询效率‌。

HashMap的线程安全性

HashMap本身是线程不安全的‌。当多个线程同时对HashMap进行操作时,可能会导致数据丢失或覆盖。例如,当一个线程正在进行插入操作时,另一个线程也可能进行插入操作,由于线程调度的原因,可能会导致第二个线程的操作覆盖第一个线程的操作,从而造成数据不一致‌。

26.final和finally和finalize的区别

27.String、StringgBuilder、StringBuffer的区别

28.重写equals已经能比较两个对象了,为什么还要重写hashcode方法

29.基本数据和包装数据类型的区别?

30.什么是多态?举个例子

31.抽象类和接口的区别

抽象类是半抽象的,有构造方法,抽象类可以包含抽象方法和非抽象方法,也可以有成员变量(不仅仅是常量)
类和类之间只能单继承,一个抽象类只能继承一个类(单继承)

接口是完全抽象的,没有构造方法,接口和接口之间支持多继承,一个类可以同时实现多个接口
这在Java 8之前基本正确,但在Java 8及之后,接口可以包含默认方法和静态方法

32.java中的几种基本类型,各占用多少字节?

33.运行时异常有哪些?

34.对面向对象的理解?面向对象有的特征有哪些?
35.HTTP报文有哪几部分组成?Get/Post有什么区别

36.转发(forward)和重定向(redirect)的区别

37.什么是AOP?什么是IOC?

38.AOP用到什么设计模式

39.SpringBean的生命周期

40.SpringIOC是怎么实现了依赖注入,有几种依赖注入方式?

41.SpringMVC的执行流程

42.SpringMVC与springboot的区别?

43.SpringMVC的事物有过哪几种?怎么处理的?

44.SpringBoot在初始化以前执行一个方法,应该怎么做?初始化以后在执行一个方法应该怎么做,有哪几种方法去做

45.在项目中会经常用到哪些注解?讲出十个

46.@Autowired注入和自己new一个对象有什么区别?

47.myBaties防sql注入的操作

48.Mybaties都要用到哪些标签

49.#和$的区别

#原样输出

$占位符

50.Mybaties如何实现循环?

51.@Component,@Controller,@Repository,@Service有何区别?

52.SpringBoot中的注解?SpringBoot的核心注解是什么?

53.springBoot要发布一个接口需要几个注解?

54.SpringBoot的运行原理/启动流程?

55.SpringBoot项目启动类上有什么注解?除了常用的还有什么注解?

56.SpringBoot配置文件怎么配?

57.SpringBoot怎么整合MyBatis

58.SpringBoot项目启动时配置文件加载顺序

59.事物的隔离级别

60.事物的七种传播方式

61.过滤器和拦截器的区别?

62.Spring运用了什么设计模式?讲一下哪些部分用到了这些设计模式

63.去重SQL语句怎么写

64.MYSQL大批量数据如何导入

65.mysql如何做大数据量分页

66.mysql索引类型

67.怎么避免索引失效

68.了解过MySQL数据库储存引擎么?

69.讲讲mysql最左匹配原则

70.笛卡尔积是什么

71.脏读、幻读、不可重复读

72.数据库的优化方案?一个SQL语句如何让他搜索变快?你们遇到的数据量最大的表是什么样,有多少条数据。

73.慢日志怎么使用?MYSQL怎么发现动作速度慢的SQL语句?

74.分库分表是怎么做的?数据库分表分库后怎么保证其一致性?

75.MySQL表为何产生死锁?如何避免死锁?

76.oracle与mysql的区别?mysql分页和oracle分页的区别?

77.Mysql默认的事务隔离级别?

78.Mysql的索引的数据结构?

79.Mysql中count()和count(1)区别

80.Sql看执行计划?

81.创建线程的方式及其区别

82.mysq1锁的粒度?并发编程锁到行还是表?(介绍页级、表级、行级)

83.什么是线程和进程?它们之间的区别是什么?

84.请解释一下线程的几种状态以及状态之间的转换关系

85.什么是线程上下文切换?

86.项目中是否用到过多线程?怎么用的?

87.Sleep()和wait()区别?Start()和run()区别?wait,sleep, notify的区别?

88.线程安全指的是什么?如何保障线程安全

89.线程锁有几种方式?使用的关键字?

90.什么是乐观锁和悲观锁?

91.synchronized和]ock的区别。Synchronize关键字修饰方法和修饰代码块的区别

92.如何保证单例模式在多线程中的线程安全性

93.线程池的常见类型石那些?

94.线程池中线程的停止?

95.线程池中的线程数多少合适?

96.线程池一般是自动创建的还是手动?为什么?

97.ReentrantLock和synchronized的别是什么

98.什么是死锁?如何避免死锁?

99.Java 中线程间通信有哪些方式


http://www.ppmy.cn/server/157260.html

相关文章

PHP相关问题练习

题目&#xff1a; <?php highlight_file(__FILE__); error_reporting(0); $val1 $_GET[val1]; $val2 $_GET[val2]; $val3 $_GET[val3]; $val4 $_GET[val4]; $val5 (string)$_POST[val5]; $val6 (string)$_POST[val6]; $val7 (string)$_POST[val7]; if( $val1 $val2…

ue5玩家角色添加武器。切换武器位置,手上武器放到背上。演示一下人体插槽和武器的连接。仅仅演示,实际项目不是这么用的

把第一人称资源包导进来 这就是我们枪的骨骼网格体 我们找到这个骨骼 右手添加插槽 取个名字 因为武器上也有动画&#xff0c;所有武器单独写个蓝图类 新建一个蓝图类 BP_Weapon 把枪的蓝图拖到人的静态网格体下&#xff0c;成为一个部分 选中BP_Weapon的父类套接字…

鸿蒙开发(29)弹性布局 (Flex)

概述 弹性布局&#xff08;Flex&#xff09;提供更加有效的方式对容器中的子元素进行排列、对齐和分配剩余空间。常用于页面头部导航栏的均匀分布、页面框架的搭建、多行数据的排列等。 容器默认存在主轴与交叉轴&#xff0c;子元素默认沿主轴排列&#xff0c;子元素在主轴方…

【HTML+CSS+JS+VUE】web前端教程-4-标签之段落、换行、水平线

标签之段落 段落是通过<p>标签定义的换行 如果您希望在不生产一个新段落的情况下进行换行(新行),请使用<br><br />元素是一个空的HTML元素 <p>这个

【Excel笔记_2】单元格跳转求累加

需求&#xff1a; 计算当前空格等于某一列1-12的和&#xff0c;下一个空格是13-24的和&#xff0c;每次间隔12 如果起始位置是 A1&#xff0c;并且希望每次计算 12 行的和&#xff0c;每个单元格依次计算不同的 12 行的和&#xff0c;可以使用以下公式&#xff1a; 在 B1 单元…

2025年伊始:回顾 ChatGPT 引发的 AI 变革以及未来展望

&#x1f4e2;&#x1f4e2;&#x1f4e2; 大家好&#xff0c;我是云楼Yunlord&#xff0c;CSDN博客之星人工智能领域前三名&#xff0c;多年人工智能学习工作经验&#xff0c;一位兴趣稀奇古怪的【人工智能领域博主】&#xff01;&#xff01;&#xff01;&#x1f61c;&#…

地市级医院人工智能应用路线图2025:机遇、挑战与战略路径(基本版附四阶段流程)

一、引言 1.1 研究背景与意义 地市级医院作为我国医疗体系的关键组成部分,承担着为广大民众提供基础与专科医疗服务的重任。伴随人口老龄化加剧、慢性病患病率上升以及民众健康需求持续增长,地市级医院面临着前所未有的医疗服务压力。据相关数据显示,近十年我国地市级医院的…

概率论常用的分布公式

01 常见离散型分布及其概率分布、期望和方差公式 伯努利分布 概率分布&#xff1a;期望&#xff1a; E(X)p方差&#xff1a;D(X)p(1−p) 二项分布 概率分布&#xff1a;期望&#xff1a;E(X)np方差&#xff1a; D(X)np(1−p) 表示方法&#xff1a;X∼B(n,p) 泊松分布 概率分布…