Java开发的多个方面,包括但不限于Java基础知识、多线程并发、JVM、框架使用、数据库、设计模式、网络编程等。
以下是一些常见的问题以及回答的方向:
Java 开发技术常见问题(一)
Java 基础知识
- 对象和类的区别是什么?
类是蓝图,对象是根据蓝图创建的具体实例。 - 什么是封装、继承和多态?
- 封装是隐藏对象的实现细节,只暴露操作的接口。
- 继承是从已有类派生新类的过程,新类能继承父类的属性和方法。
- 多态是同一个接口可以被不同的实例以不同的方式实现。
多线程并发
- 什么是线程安全?
线程安全是指在多线程环境中,代码的执行不会导致数据不一致或者程序状态异常。 - 如何保证线程安全?
可以使用synchronized关键字、Lock接口、volatile关键字、原子类等机制。
JVM
- JVM的内存模型是怎样的?
JVM内存模型包括方法区、堆、栈、程序计数器和本地方法栈。 - 什么是垃圾回收?
垃圾回收是JVM自动回收不再使用的对象所占用的内存。
框架使用
- Spring框架的核心是什么?
Spring框架的核心是控制反转(IoC)和面向切面编程(AOP)。 - 什么是MVC模式?
MVC模式将应用程序分为模型(Model)、视图(View)和控制器(Controller)三个部分。
数据库
- 什么是SQL注入?如何预防?
SQL注入是一种攻击技术,攻击者可以通过在应用程序的输入字段中插入恶意SQL代码来破坏或操纵后端数据库。
预防方法包括使用预编译的SQL语句、对输入进行验证和转义等。 - 索引是如何工作的?
索引是一种数据结构,可以快速定位到数据库表中的行,加速查询速度。
设计模式
- 什么是单例模式?
单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。 - 工厂模式和抽象工厂模式有什么区别?
工厂模式用于创建一个接口的实例,而抽象工厂模式用于创建一系列相关或依赖对象的实例。
网络编程
- TCP和UDP有什么区别?
TCP是面向连接的、可靠的传输层协议,而UDP是无连接的、不保证可靠性的协议。 - HTTP和HTTPS有什么区别?
HTTP是超文本传输协议,而HTTPS是HTTP的安全版本,通过SSL/TLS进行加密。
其他常见问题
- 什么是RESTful API?
RESTful API是一种设计风格,用于网络应用程序之间的交互,使用标准的HTTP方法。 - 如何处理分布式系统中的一致性问题?
可以使用分布式锁、事务、最终一致性模型等策略。
Java 开发技术常见问题(二)
-
HashMap实现原理:
- HashMap基于哈希表的Map接口实现,通过键(Key)的hashCode来计算索引值,将键值对存储在哈希表中。
- 当发生哈希冲突时,HashMap使用链表或红黑树(Java 8及以上版本)来解决冲突。
-
ConcurrentHashMap实现原理:
- ConcurrentHashMap允许多个线程并发访问,通过分段锁(Segment)来提高并发性能。
- 它内部由多个Segment组成,每个Segment是一个类似HashMap的结构,可以独立进行扩容和同步。
-
红黑树:
- 红黑树是一种自平衡的二叉查找树,每个节点要么是红色,要么是黑色。
- 它通过颜色和特定的规则来保持树的平衡,查找、插入和删除操作的时间复杂度为O(log n)。
-
为什么允许局部不平衡:
- 局部不平衡允许红黑树在插入和删除操作中快速重新平衡,从而保持整体的平衡性。
-
TCP和UDP的区别:
- TCP是面向连接的、可靠的传输层协议,提供数据包的顺序传输、错误检测和重传机制。
- UDP是无连接的、不可靠的传输层协议,它提供更快的数据传输速度,但不保证数据包的顺序、完整性或可靠性。
-
为什么可靠和不可靠:
- TCP可靠是因为它能确保数据包的顺序和完整性,适用于需要可靠性的应用,如网页浏览。
- UDP不可靠是因为它不保证数据包的顺序和完整性,适用于对实时性要求高的应用,如视频会议。
-
一次HTTP请求的全过程:
- 包括DNS解析,将域名转换为IP地址。
- 通过IP地址定位主机,建立TCP连接(三次握手)。
- 发送HTTP请求,服务器处理请求并返回响应。
- 客户端接收响应,关闭TCP连接。
-
TCP三次握手:
- 第一次握手:客户端发送SYN到服务器,请求建立连接。
- 第二次握手:服务器收到SYN后,回复SYN-ACK,表示同意建立连接。
- 第三次握手:客户端收到SYN-ACK后,发送ACK,完成连接建立。
-
MySQL事务:
- 事务是一系列操作,要么全部成功,要么全部失败。
- 四大特性(ACID):原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability)。
- 四大隔离级别:读未提交(Read Uncommitted)、读已提交(Read Committed)、可重复读(Repeatable Read)、串行化(Serializable)。
-
ConcurrentHashMap和Hashtable的区别:
- ConcurrentHashMap支持高并发,使用分段锁提高性能。
- Hashtable是同步的,性能较低,且不允许空键和空值。
-
Spring IOC和AOP:
- IOC(控制反转):Spring容器负责管理对象的创建和依赖注入,降低组件间的耦合。
- AOP(面向切面编程):允许将横切关注点(如日志、事务)与业务逻辑分离。
- 优点:IOC简化了对象创建和管理,AOP提高了代码的模块化和可维护性。
-
常用的线程池:
- 固定大小的线程池(FixedThreadPool)。
- 缓存线程池(CachedThreadPool)。
- 单线程线程池(SingleThreadExecutor)。
- 调度线程池(ScheduledThreadPool)。
-
Runnable和Thread创建线程:
- Runnable是任务接口,可以被Thread执行。
- Thread是线程类,可以执行Runnable任务。
- 使用Runnable可以更好地实现多线程,因为它允许多个线程共享同一个任务。
-
Callable和Runnable的区别:
- Callable可以返回值和抛出异常,而Runnable不能。
- Callable通常与ExecutorService一起使用。
-
线程方法中的异常处理:
- 线程方法中的异常需要在方法内部捕获和处理。
- 主线程通常无法捕获到子线程中的异常,除非通过特定的机制(如Future)。
-
synchronized和锁的区别:
- synchronized是Java的关键字,用于同步方法或代码块。
- 锁(如ReentrantLock)是JDK提供的锁接口,提供了更多的灵活性和高级功能。
- synchronized使用简单,但功能有限;ReentrantLock功能强大,但使用复杂。
-
JVM的对象分配:
- 对象通常在堆(Heap)内存中分配。
- 类加载器在加载类时,Class对象会被分配到方法区(在Java 8之前)或元空间(Java 8及之后)。
Java 开发技术常见问题(三)
-
常用的设计模式介绍:
- 单例模式:确保一个类只有一个实例,并提供一个全局访问点。
- 装饰者模式:动态地给一个对象添加额外的职责,而不改变其结构。
-
Java会出现内存溢出吗?什么情况下会出现?
- 是的,Java也会出现内存溢出,特别是当JVM堆内存被占满且无法回收时。
-
双亲委派模型:
- 双亲委派模型是Java类加载机制,确保Java核心库的安全性和一致性。
- 它要求除了顶层的启动类加载器外,其余的类加载器都应先委托其父加载器试图加载某个类。
-
对象什么情况下进入老年代?
- 对象在新生代的Eden区和Survivor区之间经过多次GC后,如果仍然存活,会被晋升到老年代。
-
快速排序过程:
- 选择一个基准值(pivot)。
- 将数组分为两部分,一部分包含比基准值小的元素,另一部分包含比基准值大的元素。
- 递归地在这两部分上重复这个过程。
-
AOP实现原理:动态代理:
- 动态代理可以在运行时动态地创建代理对象,拦截方法调用,并在方法调用前后添加额外的行为。
-
BIO、NIO、AIO:
- BIO(阻塞I/O):每个连接都需要一个单独的线程处理,适合连接数较少的场景。
- NIO(非阻塞I/O):通过缓冲区和通道的概念,可以处理更多的并发连接,且线程数不变。
- AIO(异步I/O):进一步抽象,允许应用程序发出I/O请求而不必等待I/O操作完成。
-
消息中间件有哪些?他们之间的优劣势?
- 常见的消息中间件有Kafka、RabbitMQ、ActiveMQ等。
- Kafka适合高吞吐量的场景,RabbitMQ支持多种消息协议,ActiveMQ是JMS规范的实现。
-
Redis,持久化框架:
- Redis支持RDB和AOF两种持久化方式。
- RDB是快照方式,AOF是日志方式,记录每次写操作。
-
栈和队列:
- 栈:后进先出(LIFO)的数据结构。
- 队列:先进先出(FIFO)的数据结构。
-
垃圾回收算法:
- 标记-清除:标记需要回收的对象,然后清除它们。
- 复制算法:将内存分为两个区域,复制存活的对象到另一个区域,然后清除原区域。
- 标记-整理:在标记-清除的基础上,移动存活的对象,减少内存碎片。
-
MySQL的索引:
- 索引用于快速查找、排序和分组数据。
- 常见的索引类型有B树索引、哈希索引、全文索引等。
-
Tomcat 类加载器:
- Tomcat使用类加载器隔离每个Web应用,确保应用之间的独立性。
-
OOM内存泄漏,什么情况下会出现,如何排查:
- OOM(Out of Memory)当JVM堆内存耗尽时出现。
- 内存泄漏通常由于长时间持有对象引用导致。
- 排查内存泄漏可以使用工具如jmap、VisualVM等,分析堆转储文件。