《字节码到JVM:Java基础核心知识点全解析(小林八股·上)》 个人主页北极的代码欢迎来访作者简介java后端学习者❄️个人专栏苍穹外卖日记SSM框架深入JavaWeb✨命运的结局尽可永在不屈的挑战却不可须臾或缺前言最近刚学完黑马点评可以说是休整一段时间也可以说是摆了一段时间吧虽然知道还有很多需要学习的这周要开始努力了整理一下点评算法和八股同步进行然后再看看agent相关的知识今天上课的时候看了java基础八股的一些知识这里总结一下吧适合回顾面试重点虽然不难但是也很重要所谓基础不牢地动山摇。摘要本文总结了Java核心知识点包括跨平台原理JVM机制、JVM/JRE/JDK关系、值传递特性。重点解析面向对象三大特性封装、继承、多态及重载与重写的区别对比抽象类和接口的应用场景。详细探讨数据类型基本类型与包装类缓存池、字符串处理String/StringBuilder/StringBuffer及浮点数精度问题解决方案。最后对比了/equals()/hashCode()的差异分析成员/局部变量存储区别以及深浅拷贝实现方式。适合Java学习者巩固基础知识和面试准备。一、 Java核心特性与运行机制1. 为什么Java能做到一次编写到处运行这得益于Java的编译与解释并存的机制。Java源代码.java首先被javac编译器编译成字节码.class字节码不是底层机器码而是面向Java虚拟机的中间码。随后Java虚拟机JVM充当翻译官将字节码解释或编译通过JIT即时编译器成当前操作系统能识别的机器码。跨平台的是Java程序而不是JVM——不同平台需要安装不同版本的JVM。2. JVM、JRE、JDK 三者的关系“套娃”结构从内到外看JVM (Java虚拟机)核心运行环境执行字节码。JRE (Java运行时环境)JVM 核心类库如rt.jar。不能创建新程序。JDK (Java开发工具包)JRE 开发工具如javacjarjconsole。能编译和运行。3. Java是值传递还是引用传递结论Java只有值传递。基本类型传递的是值的副本。修改副本不影响原变量。引用类型传递的是地址值的副本也就是引用的拷贝。这意味着形参和实参指向堆中同一个对象所以通过形参修改对象的属性会影响原对象但如果将形参指向一个新对象重新赋值原对象的引用依然指向原来的地址不会受影响。二、 面向对象核心OOP1. 三大特性封装、继承、多态封装将属性私有化private提供公共的getter/setter方法访问。目的是保护数据完整性隐藏实现细节。继承子类继承父类复用代码。Java中是单继承但支持多层继承。多态父类引用指向子类对象。如List list new ArrayList();。多态的必要三要素继承、重写、向上转型。编译看左边检查是否有该方法运行看右边执行子类重写的方法。2. 重载Overload vs. 重写Override特性重载 (Overload)重写 (Override)范围同一个类中父子类中方法名必须相同必须相同参数列表必须不同个数、类型、顺序必须相同返回值无关可以不同返回值类型需是父类返回类型的子类或相同访问修饰符无关权限不能更严格如父类public子类不能是private抛出异常无关异常范围不能更大或更少绑定编译期多态静态绑定运行期多态动态绑定3. 抽象类 vs. 接口抽象类是is-a关系用于描述事物的本质。可以有构造方法、普通成员变量、具体方法有实现。子类用extends继承是单继承。接口是like-a关系用于定义规范/行为。JDK8后允许有default和static具体方法JDK9后允许private方法。变量默认是public static final。实现类用implements实现可以多实现。三、 数据类型与字符串细节1. 基本数据类型 vs. 包装类型存储局部变量基本类型存储在栈中包装类型对象存储在堆中。但如果有JIT编译器逃逸分析发现对象未逃逸可能会进行标量替换分配到栈上。比较基本类型用比较数值包装类型比较内容用equals()比较地址用。用途POJO实体类中的属性必须用包装类型因为数据库查询结果可能为null基本类型无法表达null。2. 包装类型的缓存池常量池大部分包装类都利用了享元模式。IntegerShortLongByte默认缓存了[-128 127]之间的数据。调用valueOf()时会直接返回缓存对象而不是新建。Character缓存[0 127]。Boolean缓存TRUE/FALSE。Double、Float没有缓存。注意Integer a 100; Integer b 100;此时a b为true若值为 200则为false。3. 浮点数精度丢失与解决方案原因十进制小数转换为二进制时乘2取整可能产生无限循环如0.2而浮点数float/double的存储位数有限IEEE 754标准必须截断因此产生精度误差。解决方案使用BigDecimal。注意必须使用 String 参数的构造器即new BigDecimal(0.1)而不是new BigDecimal(0.1)后者依然会带入二进制精度误差。比较大小用compareTo()而非equals()因为equals会比较精度例如1.0与1.00。java // 错误示例 double d 2.0 - 1.1; // 结果: 0.8999999999999999 // 正确示例 BigDecimal b1 new BigDecimal(2.0); BigDecimal b2 new BigDecimal(1.1); BigDecimal result b1.subtract(b2); // 结果: 0.9 // 比较 System.out.println(b1.equals(new BigDecimal(2.00))); // false (精度不同) System.out.println(b1.compareTo(new BigDecimal(2.00))); // 0 (视为相等)4. String、StringBuilder、StringBufferString不可变final修饰任何修改都会产生新对象。线程安全天生。适合少量字符串操作或常量。StringBuilder可变线程不安全无锁性能最高。首选用于单线程下的字符串拼接如循环中的操作会被编译器优化为StringBuilder。StringBuffer可变线程安全方法加synchronized性能较低。适合多线程环境下的字符串操作。底层实现父类AbstractStringBuilder中使用byte[]数组存储JDK9以后之前是char[]。初始化容量为字符串长度 16。扩容时通常扩容为原容量的2倍 2会触发数组拷贝开销较大建议预分配大小。四、 对比与进阶1.vsequals()vshashCode()对于基本类型比较值对于引用类型比较内存地址。equals()默认比较地址即等价于但类可以重写它如String、Integer来比较内容。hashCode() 契约两个对象相等equals为 true则hashCode必须相等。两个对象hashCode相等equals不一定为 true哈希冲突。重写 equals 必须重写 hashCode否则在HashSet/HashMap中equals相等的两个对象因为hashCode不同会被存储在两个不同的桶里导致逻辑混乱存不进去或取不出来。2. 成员变量 vs 局部变量成员变量属于类或实例存储在堆中。有默认值int默认0对象默认null生命周期随对象。局部变量定义在方法内存储在栈中。无默认值必须显式初始化生命周期随方法调用结束而销毁。3. 深拷贝 vs 浅拷贝引用拷贝只是复制了指针两个引用指向同一个对象。浅拷贝复制对象本身但如果对象内部有引用其他对象只复制引用不复制内部对象。实现Cloneable接口并重写clone()方法默认是浅拷贝。深拷贝不仅复制对象还会递归复制其内部引用的所有对象。可通过重写clone()手动为引用类型调用clone()或通过序列化JSON/IO方式实现。4. 静态方法为什么不能调用非静态成员生命周期原因。静态方法属于类在类加载时就已经存在此时可能还没有创建任何对象实例。非静态成员属于对象只有在实例化后才在内存中分配。在静态方法中直接调用非静态成员就像是访问一个还不存在的东西编译器会报错。附录Object类常用方法速查equals()hashCode()toString()getClass()clone()wait()notify()/notifyAll()finalize()。