2024面试offer收割宝典字节篇
1.IO 模型有哪些,讲讲你理解的 nio ,他和 bio,aio 的区别是啥, 谈谈 reactor 模型。
IO 模型主要包括以下几种:
1. 阻塞 I/O (BIO):
- 当一个线程调用 read() 或 write() 系统调用时,如果数据没有准备好或者缓冲区已满,则该线程会被操作系统阻塞,直到有数据可读或写入完成。
- 在Java中传统的java.io包中的类如Socket和ServerSocket就是基于BIO模型实现的。
2. 非阻塞 I/O (NIO):
- Java NIO(New Input/Output)在Java 1.4版本引入,提供了非阻塞的I/O能力。
- 使用NIO,当调用channel.read()或channel.write()方法时,若当前无数据可读或缓冲区满不能立即写入,不会阻塞线程,而是会返回一个状态值(通常是0,表示没有数据可读或没有足够的空间写入)。
- NIO支持单线程处理多个Channel(通道),通过Selector可以监听多个Channel的事件,一旦某个Channel准备就绪进行读写操作,Selector就会通知相应的线程去处理。
- NIO的核心组件包括Buffer(缓冲区)、Channel(通道)和Selector(选择器)。
3. 异步 I/O (AIO, NIO.2 或 AIO in Java):
- 异步非阻塞I/O在Java 7中引入,也被称为NIO.2。
- 在AIO模式下,当发起一个读写请求后,应用程序可以继续执行其他任务,而无需等待操作结果。
- 当数据准备完毕或操作完成时,操作系统会回调预先注册的CompletionHandler(完成处理器)来处理数据或处理完成后的逻辑。
- 这种模型极大地提高了系统并发处理能力和资源利用率,特别适合于高并发、低延迟的应用场景。
Reactor 模型是一种设计模式,常用于处理大量并发I/O请求。它通常配合NIO或AIO使用,在服务器端能够同时监听多个连接请求,并且根据不同的事件类型分发给相应的处理器进行处理。基本结构包括:
- Dispatcher:负责监听并分发事件到对应的Handler。
- Handlers:处理实际的I/O事件,如读取数据、写入响应等。
- Event Demultiplexer:在Java NIO中对应的是Selector,它可以监控多个Channel上的事件,并在事件发生时通知Dispatcher。
总结来说,BIO、NIO与AIO的主要区别在于线程模型和阻塞行为上:
- BIO是一个连接一个线程,容易造成线程资源浪费,性能瓶颈明显。
- NIO在一个线程内可以管理多个连接,通过非阻塞方式减少无效等待时间,但程序需要自行轮询检查数据是否准备好。
- AIO则更进一步,由操作系统直接负责数据的读写操作,并在完成后通知应用,从而避免了不必要的上下文切换,理论上能提供最高的并发性能。
2.反射的原理,反射创建类实例的三种方式是什么。
反射是Java中的一项重要特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的各种属性和功能。
在运行时,Java反射机制可以加载类、获取类的信息(如类名、方法、字段等)、实例化对象以及调用方法等。
反射创建类实例的三种主要方式如下:
1. 通过无参构造函数创建:
// 假设我们有一个名为Book的类Class<?> bookClass = Class.forName("com.example.Book");Constructor<?> constructor = bookClass.getConstructor(); // 获取无参构造器Book book = (Book) constructor.newInstance(); // 调用无参构造器创建实例
2. 通过带参数的构造函数创建:
Class<?> bookClass = Class.forName("com.example.Book");Constructor<Book> constructor = bookClass.getDeclaredConstructor(String.class, String.class); // 获取带有String类型参数的私有构造器constructor.setAccessible(true); // 如果构造器是私有的,需要设置为可访问Book book = constructor.newInstance("The Title", "The Author"); // 使用指定参数创建实例
3. 通过Class的newInstance()方法创建:
// 注意:这种方式仅适用于具有默认(无参或public)构造函数的类Class<Book> bookClass = Book.class;try {Book book = bookClass.newInstance();} catch (InstantiationException | IllegalAccessException e) {// 处理异常,例如如果类是抽象类或者没有默认构造函数e.printStackTrace();}
上述代码展示了如何通过反射API动态地根据类信息创建类的实例。需要注意的是,对于非public的构造函数或其他成员,可能需要调用setAccessible(true)来绕过访问控制检查。
3.反射中,Class.forName 和 ClassLoader 区别 。
在Java反射机制中,Class.forName() 方法和 ClassLoader 类加载器都用于动态加载类,但它们之间存在一些区别:
1. Class.forName():
- 这是一个静态方法,位于 java.lang.Class 类中。
- 它接受一个字符串参数,该参数是完全限定类名(包括包名)。
- 当调用 Class.forName(className) 时,它会按照当前线程的上下文类加载器(context class loader)自动寻找并加载指定名称的类,并初始化该类。如果类还未被加载过,则执行类加载过程,包括验证、准备、解析和初始化阶段。
- 如果类中有静态初始化块或静态字段会被初始化。
- 通常抛出 ClassNotFoundException,如果找不到指定的类。
2. ClassLoader.loadClass() 或 ClassLoader.findSystemClass() 等相关方法:
- ClassLoader 是 Java 中用来加载类的抽象类,每个类在运行时都有一个与之关联的类加载器。
- 使用自定义的 ClassLoader 实例或者系统的默认类加载器,可以更灵活地控制类的加载过程,例如实现自己的类加载逻辑、支持热加载等高级功能。
- 调用如 ClassLoader.loadClass(className) 不会立即初始化类,仅完成类的加载过程,即验证、准备、解析阶段,而不会执行类的初始化(构造静态变量和执行静态初始化块)。
- 可以通过调用 Class.forName(className, initialize, classLoader) 来控制是否进行初始化操作,其中 initialize 参数为布尔值。
- 同样可能抛出 ClassNotFoundException。
总结来说,两者的主要区别在于:
- Class.forName() 默认情况下包含了完整的类加载和初始化过程,更加直观易用,常用于简单场景下的类加载。
- ClassLoader 提供了更底层和灵活的类加载能力,可以根据需要定制加载策略,且默认情况下不进行类的初始化。
4.描述动态代理的几种实现方式,分别说出相应的优缺点。
在Java中,动态代理有两种主要的实现方式:JDK动态代理和CGLIB代理。
1. JDK动态代理:
- 原理:基于Java反射机制实现。它通过InvocationHandler接口创建一个代理类,该代理类在运行时继承自Proxy类,并实现了与目标对象相同的接口。
- 优点:
- 由于是基于接口的,所以可以很方便地为一组具有相同接口的目标对象生成代理,适用范围广。
- 实现代理逻辑比较简单,只需要实现InvocationHandler接口即可。
- 不需要预先知道所有要代理的方法,代理逻辑更加灵活。
- 缺点:
- 需要目标类实现接口,对于没有接口的类无法进行代理。
- 由于使用了反射,在性能上相比于直接调用有一定损耗。
- 功能相对简单,不能针对类的成员变量进行操作。
2. CGLIB代理:
- 原理:基于ASM字节码生成框架,能够在运行期对字节码进行修改或动态生成新的类。CGLIB代理生成的代理类是被代理类的一个子类。
- 优点:
- 不依赖于接口,能够实现代理没有实现任何接口的目标类,适用性更广泛。
- 相比于JDK动态代理,CGLIB代理功能更加强大,可以拦截方法、修改方法等。
- 缺点:
- CGLIB代理会对类的所有方法进行增强,即使不希望增强的方法也会被代理,这可能造成额外的性能开销。
- 如果目标类做了final修饰或者有final方法,则无法进行代理。
- CGLIB库不是Java标准库的一部分,需要单独引入依赖。
- 字节码操作技术相比反射更为复杂,如果出现问题调试起来较为困难。
总结来说,JDK动态代理更适合接口较多且不需要对非接口方法进行增强的情况,而CGLIB代理则在不需要接口或需要对类的所有方法进行增强时更加适用。在实际应用中,如Spring框架会根据目标对象是否有接口智能选择使用哪种动态代理策略。
5.动态代理与 cglib 实现的区别。
JDK动态代理实现:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;public class JdkProxyExample {interface MyInterface {void doSomething();}static class TargetClass implements MyInterface {@Overridepublic void doSomething() {System.out.println("Target: Doing something...");}}static class LoggingInvocationHandler implements InvocationHandler {private final Object target;public LoggingInvocationHandler(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("Before method call: " + method.getName());Object result = method.invoke(target, args);System.out.println("After method call: " + method.getName());return result;}public static MyInterface createLoggingProxy(TargetClass target) {return (MyInterface) Proxy.newProxyInstance(TargetClass.class.getClassLoader(),new Class<?>[]{MyInterface.class},new LoggingInvocationHandler(target));}}public static void main(String[] args) {MyInterface target = new TargetClass();MyInterface proxy = LoggingInvocationHandler.createLoggingProxy(new TargetClass());proxy.doSomething(); // 输出日志并调用目标方法}
}
在上述例子中,LoggingInvocationHandler实现了InvocationHandler接口,当代理对象的方法被调用时,会触发invoke方法,在该方法中添加了前后打印日志的逻辑,并实际调用了目标类的方法。
CGLIB代理实现:
首先需要引入CGLIB库(例如使用Spring框架中的CGLIB代理支持):
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;public class CglibProxyExample {static class TargetClass {public void doSomething() {System.out.println("Target: Doing something...");}}static class LoggingMethodInterceptor implements MethodInterceptor {@Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("Before method call: " + method.getName());Object result = proxy.invokeSuper(obj, args); // 调用父类(即目标类)的方法System.out.println("After method call: " + method.getName());return result;}public static TargetClass createLoggingProxy(TargetClass target) {Enhancer enhancer = new Enhancer();enhancer.setSuperclass(TargetClass.class);enhancer.setCallback(new LoggingMethodInterceptor());return (TargetClass) enhancer.create();}}public static void main(String[] args) {TargetClass target = new TargetClass();TargetClass proxy = LoggingMethodInterceptor.createLoggingProxy(new TargetClass());proxy.doSomething(); // 输出日志并调用目标方法}
}
在这个CGLIB代理的例子中,我们创建了一个LoggingMethodInterceptor类,它实现了MethodInterceptor接口。当代理对象的方法被调用时,会触发intercept方法,在这个方法里同样添加了日志输出,并通过MethodProxy调用了目标类的实际方法。CGLIB库通过字节码增强技术动态生成了继承自TargetClass的新类作为代理对象。
相关文章:

2024面试offer收割宝典字节篇
1.IO 模型有哪些,讲讲你理解的 nio ,他和 bio,aio 的区别是啥, 谈谈 reactor 模型。 IO 模型主要包括以下几种:1. 阻塞 I/O (BIO): 当一个线程调用 read() 或 write() 系统调用时,如果数据没有准备好或者缓冲区已满,则该线程会被操作系统阻塞,直到有数据可读或写入完…...

冒泡排序及其优化
冒泡排序 int[] arr {1,3,2,9,4,7,2,8};//比较多少轮(n个数字比较n-1次)for(int i0,n arr.length;i<n-1;i) {//每轮比较多少次(n-1-i次)for(int j 0;j<n-1-i;j) {//两两比较if(arr[j] > arr[j1]) { //比较结果为升序排列,如果想要降序排列结果将 >…...

【医学大模型 补全主诉】BioGPT + LSTM 自动补全医院紧急部门主诉
BioGPT LSTM 自动补全医院紧急部门主诉 问题:针对在紧急部门中自动补全主诉的问题子问题1: 提高主诉记录的准确性子问题2: 加快主诉记录的速度子问题3: 统一医疗术语的使用子问题4: 减少打字错误和误解子问题5: 提高非特定主诉的处理能力 解法数据预处理神经网络方…...

HCIE-Datacom证书有效期多久?HCIE考试有哪些内容?
如今越来越多的人开始关注并参与到华为认证的学习中来。 其中,华为认证数据通信专家(HCIE-Datacom)作为华为认证体系中的高级认证,备受瞩目。 那么,关于HCIE-Datacom证书的有效期以及HCIE考试的内容,你知道多少呢?下…...

OpenCV中的边缘检测技术及实现
边缘检测是在电脑如何理解图片这一问题中的一环,它帮助电脑找出照片里的轮廓和分界线。想象一下你在看一幅黑白漫画,轮廓线定义了每一个角色和物体,而电脑要做的,就是通过边缘检测来找出这些线条。这在很多像是图像分析这样的领域…...
机器学习基础(一)理解机器学习的本质
导读:在本文中,将深入探索机器学习的根本原理,包括基本概念、分类及如何通过构建预测模型来应用这些理论。 目录 机器学习 机器学习概念 相关概念 机器学习根本:模型 数据的语言:特征与标签 训练与测试…...

Eclipse - Makefile generation
Eclipse - Makefile generation References right mouse click on the project -> Properties -> C/C Build -> Generate Makefiles automatically 默认会在 Debug 目录下创建 Makefile 文件。 References [1] Yongqiang Cheng, https://yongqiang.blog.csdn.net/...

Sora:新一代实时音视频通信框架
一、Sora简介 Sora是一个开源的实时音视频通信框架,旨在提供高效、稳定、可扩展的音视频通信解决方案。它基于WebRTC技术,支持跨平台、跨浏览器的实时音视频通信,并且具备低延迟、高并发、易集成等特点。 --点击进入Sora(一定要科学哦&#x…...

龟兔赛跑算法
一、题目 给定一个长度为 n1 的数组nums,数组中所有的数均在 1∼n1 的范围内,其中 n≥1。 请找出数组中任意一个重复的数。 样例 给定 nums [2, 3, 5, 4, 3, 2, 6, 7]。返回 2 或 3。 二、解析 解决这个问题的一种有效方法是使用快慢指针…...

Yii2项目使用composer异常记录
问题描述 在yii2项目中,使用require命令安装依赖时,出现如下错误提示 该提示意思是:composer运行时,执行了yiisoft/yii2-composer目录下的插件,但是该插件使用的API版本是1.0,但是当前的cmposer版本提供的…...

【蓝桥杯 2021】图像模糊
图像模糊 题目描述 小蓝有一张黑白图像,由 nm 个像素组成,其中从上到下共 n 行,每行从左到右 m 列。每个像素由一个 0 到 255 之间的灰度值表示。 现在,小蓝准备对图像进行模糊操作,操作的方法为: 对于…...

【leetcode】贪心算法介绍
详细且全面地分析贪心算法常用的解题套路、数据结构和代码逻辑如下: 找最值型: 每一步选择都是局部最优解,最后得到的结果就是全局最优解。常用于找零钱问题、区间覆盖问题等。一般情况下,可以通过排序将数据进行处理,…...

com.alibaba.fastjson.JSONException: toJSON error的原因
问题: 导出接口报错,显示json格式化异常 发现问题: 第一个参数为HttpResponse,转换成json的时候报错 修改方法: 1.调换两个参数的位置 2.在aop判断里边 把ServletAPI过滤掉 Before("excudeWebController()")pub…...

华为配置旁挂二层组网直接转发示例
配置旁挂二层组网直接转发示例 组网图形 图1 配置旁挂二层组网直接转发示例组网图 业务需求组网需求数据规划配置思路配置注意事项操作步骤配置文件扩展阅读 业务需求 企业用户通过WLAN接入网络,以满足移动办公的最基本需求。且在覆盖区域内移动发生漫游时ÿ…...

OLMo 以促进语言模型科学之名 —— OLMo Accelerating the Science of Language Models —— 全文翻译
OLMo: Accelerating the Science of Language Models OLMo 以促进语言模型科学之名 摘要 语言模型在自然语言处理的研究中和商业产品中已经变得无所不在。因为其商业上的重要性激增,所以,其中最强大的模型已经闭源,控制在专有接口之中&#…...

单例模式双端检测详解
正确写出doublecheck的单例模式_double check单例模式-CSDN博客...

秦PLUS荣耀版7.98万元起震撼上市,拉开“电比油低”大幕
2月19日,秦PLUS荣耀版正式上市,五大颠覆、三大焕新刷新A轿体验新高度。DM-i版本5款车型,官方指导价7.98万元——12.58万元;EV版本5款车型,官方指导价10.98万元——13.98万元。正式开启“电比油低”新时代。 电比油低&a…...

学习总结19
# 奶牛的耳语 ## 题目描述 在你的养牛场,所有的奶牛都养在一排呈直线的牛栏中。一共有 n 头奶牛,其中第 i 头牛在直线上所处的位置可以用一个整数坐标 pi(0< pi < 10^8) 来表示。在无聊的日子里,奶牛们常常在自己的牛栏里与其它奶牛交…...

rancher v2.8.1 如何成功注册已有 k8s 集群
需要加入的集群为rke2部署的双节点集群 $ kubectl get node NAME STATUS ROLES AGE VERSION rke-master01 Ready control-plane,etcd,master,worker 94d v1.26.8rke2r1 rke-master02 Ready control-plane,etcd,mast…...

Vue中$root的使用方法
查看本专栏目录 关于作者 还是大剑师兰特:曾是美国某知名大学计算机专业研究生,现为航空航海领域高级前端工程师;CSDN知名博主,GIS领域优质创作者,深耕openlayers、leaflet、mapbox、cesium,canvas&#x…...

redis 异步队列
//produceMessage.ts 模拟生产者 import Redis from ioredis; const redis new Redis(); // 生产者:将消息推送到队列 async function produceMessage(queueName:string, message:string) {try {await redis.rpush(queueName, message);console.log(Produced messa…...

SpringBoot + Nacos 实现动态化线程池
1.背景 在后台开发中,会经常用到线程池技术,对于线程池核心参数的配置很大程度上依靠经验。然而,由于系统运行过程中存在的不确定性,我们很难一劳永逸地规划一个合理的线程池参数。 在对线程池配置参数进行调整时,一…...

《Docker极简教程》--Dockerfile--Dockerfile的基本语法
Dockerfile是一种文本文件,用于定义Docker镜像的内容和构建步骤。它包含一系列指令,每个指令代表一个构建步骤,从基础镜像开始,逐步构建出最终的镜像。通过Dockerfile,用户可以精确地描述应用程序运行环境的配置、依赖…...

css中, grid-auto-rows: 怎样简写在grid:中
grid-auto-rows:100px; grid-template-columns:1fr 1fr; 👆可以写成👇 grid:auto-flow 100px / 1fr 1fr;在CSS Grid布局中,grid-auto-rows 属性用于指定自动生成的网格容器的行的大小。如果你想要将 grid-auto-rows 的值简写在 grid 属性中&a…...

@ 代码随想录算法训练营第8周(C语言)|Day53(动态规划)
代码随想录算法训练营第8周(C语言)|Day53(动态规划) Day50、动态规划(包含题目 ● 123.买卖股票的最佳时机III ● 188.买卖股票的最佳时机IV ) 123.买卖股票的最佳时机III 题目描述 给定一个数组 price…...

算法-矩阵置零
1、题目来源 73. 矩阵置零 - 力扣(LeetCode) 2、题目描述 给定一个 m x n 的矩阵,如果一个元素为 0 ,则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。 示例 1: 输入:matrix [[1,1,1],[1,0,1…...

xilinx除法器的使用
平台:Vivado2018.3. 芯片:xcku115-flva1517-2-i (active) 最近学习使用了xilinx除法器,在使用过程中出现了很多次除法器的结果和我预计的结果不一致,特此记录学习一下。 参考文件:pg151.下载地址 pg151-div-gen.pdf …...

算法沉淀——递归(leetcode真题剖析)
算法沉淀——递归 01.汉诺塔问题02.合并两个有序链表03.反转链表04.两两交换链表中的节点05.Pow(x, n) 递归是一种通过调用自身的方式来解决问题的算法。在递归算法中,问题被分解为更小的相似子问题,然后通过对这些子问题的解进行组合来解决原始问题。递…...

BERT模型中的input_ids和attention_mask参数
一、概述 1.1 input_ids 在BERT模型及其衍生体中,输入文本首先经过一个分词处理流程,其中文本被细分为单词或子单词(subwords),每个分词随后映射到一个唯一的整数标识符。这些标识符组成了所谓的input_ids数组&#x…...

java+vue_springboot企业设备安全信息系统14jbc
企业防爆安全信息系统采用B/S架构,数据库是MySQL。网站的搭建与开发采用了先进的java进行编写,使用了vue框架。该系统从三个对象:由管理员、人员和企业来对系统进行设计构建。主要功能包括:个人信息修改,对人员管理&am…...