JAVA重点基础、进阶知识及易错点总结(1)---数据类型、运算符、流程控制
Java 巩固进阶 · 第1天主题数据类型、运算符与流程控制 —— 避开那些“隐形”的坑 进度概览重启Java基础。 核心价值很多生产环境的Bug如金额精度丢失、空指针崩溃、逻辑穿透都源于对这些“简单”知识的误解。今天我们将深入底层建立防御性编程思维。一、基本数据类型与字面量规范1. 八大基本类型全景图分类类型字节范围/特点默认后缀整数byte1-128 ~ 127-short2-32,768 ~ 32,767-int4最常用约±21亿(默认)long8超大整数L(推荐大写)浮点float4单精度易丢失精度f/Fdouble8默认小数类型双精度(默认)字符char2单字符Unicode (单引号)布尔boolean-true/false-2. ⚠️ 字面量陷阱与规范核心规则编译器对数字字面量的“默认偏见”。整数默认是int若赋值给long且超出int范围必须加L。小数默认是double若赋值给float必须加f否则报错“可能损失精度”。❌ 经典错误案例// 1. 浮点赋值错误floatprice9.9;// ❌ 编译报错9.9是double转float会丢精度// 2. 长整型溢出longbigNum2147483648;// ❌ 编译报错字面量被视为int已超出int最大值✅ 正确写法 最佳实践// 1. 显式后缀floatprice9.9f;// ✅ 推荐longbigNum2147483648L;// ✅ 推荐L必须大写(小写l易混淆为1)// 2. 金融铁律严禁使用 float/double 计算金额// ❌ 错误double money 0.1 0.2; // 结果可能是 0.30000000000000004// ✅ 正确使用 BigDecimalBigDecimalamountnewBigDecimal(0.1).add(newBigDecimal(0.2));二、浮点数精度陷阱为什么0.1 0.2 ! 0.31. 底层原理 (IEEE 754)计算机使用二进制存储小数。十进制的0.1在二进制中是无限循环小数(0.00011001100...)。由于存储空间有限float 32位/double 64位必须进行截断从而产生微小的精度误差。2. 致命案例doublea0.1;doubleb0.2;doublec0.3;// ❌ 经典面试题输出 falseSystem.out.println(abc);// 真相0.30000000000000004 ! 0.33. 解决方案方案 A误差范围比较 (科学计算)doubleepsilon1e-6;// 允许的最大误差if(Math.abs((ab)-c)epsilon){System.out.println(视为相等);}方案 BBigDecimal (金融/商业计算 - 唯一推荐)// ⚠️ 关键构造函数必须传 String不能传 doubleBigDecimalbd1newBigDecimal(0.1);BigDecimalbd2newBigDecimal(0.2);if(bd1.add(bd2).equals(newBigDecimal(0.3))){System.out.println(金额匹配);}三、包装类、自动装箱与 Integer 缓存池这是大厂面试最高频考点也是生产环境 NPE (空指针) 的重灾区。1. 核心机制自动装箱Integer i 10;→\rightarrow→Integer.valueOf(10)自动拆箱int n i;→\rightarrow→i.intValue()Integer 缓存池 (Integer Cache)JVM 为了优化性能预先创建了-128 到 127之间的整数对象。在此范围内valueOf()返回缓存对象引用相同。超出范围每次new一个新对象引用不同。2. ⚠️ 陷阱演示Integera127;Integerb127;System.out.println(ab);// ✅ true (指向同一个缓存对象)Integerc128;Integerd128;System.out.println(cd);// ❌ false (两个不同的对象地址不同)// ⚠️ 拆箱导致的 NPE (空指针异常)Integerenull;// 下面这行代码等价于 e.intValue()直接崩intfe;// throws NullPointerException3. 扩展知识 (面试加分项)为什么是 -128~127对应byte的范围覆盖了最常用的整数区间。其他缓存Boolean(TRUE/FALSE),Character(\u0000~\u007F),Long/Short(-128~127)。配置调整可通过 JVM 参数-XX:AutoBoxCacheMaxsize扩大 Integer 缓存上限。4. ️ 防御性编程安全拆箱数据库查询返回值常为null直接拆箱极其危险。publicclassSafeUnboxingDemo{publicvoidprocessScore(IntegerdbScore){// 【方案 A】传统三元运算符 (最快)intscoreA(dbScorenull)?0:dbScore;// 【方案 B】Optional 链式调用 (优雅适合复杂逻辑)intscoreBOptional.ofNullable(dbScore).filter(s-s0)// 过滤负分.orElse(0);// 默认值// 【方案 C】严格模式 (数据异常直接阻断防止脏数据扩散)intscoreCOptional.ofNullable(dbScore).orElseThrow(()-newIllegalArgumentException(积分不能为空));}} 最佳实践对象比较永远用.equals()拆箱前必判空Optional主要用于局部变量处理不建议作为实体类字段。四、逻辑运算符短路机制 (vs)1. 核心区别运算符名称行为特征典型场景短路与左假则右不执行判空保护(obj ! null obj.isValid())非短路/按位与左右都执行位运算、需要副作用执行的场景****短路或|非短路/按位或左右都执行位运算2. ⚠️ 生死攸关的案例Stringusernull;// ✅ 安全user ! null 为 false 短路后面不执行不会报 NPEif(user!nulluser.length()5){...}// ❌ 崩溃 强制执行右边user.length() 抛出 NullPointerExceptionif(user!nulluser.length()5){...}3. 副作用陷阱intx0;// 使用 条件为假x 被跳过if(false(x0)){}System.out.println(x);// 输出 0// 使用 无论左边如何x 都会执行if(false(x0)){}System.out.println(x);// 输出 1五、Switch 语句的进化史1. 版本演进JDK 1.4-仅支持byte,short,char,int(及枚举)。JDK 1.7支持String(底层通过hashCode()equals()实现)。JDK 14 (现代语法)支持箭头语法-(无需break天然防穿透)。支持Switch 表达式(可直接返回值)。支持多 Case 合并(case A, B - ...)。2. 代码对比 传统写法 (JDK 8及以前) - 风险忘记 breakStringtypeA;intresult0;switch(type){caseA:result1;break;// ⚠️ 忘了写 break 会导致穿透到 case BcaseB:result2;break;default:result0;} 现代写法 (JDK 14) - 推荐StringtypeA;// 直接作为表达式返回值无穿透风险代码极简intresultswitch(type){caseA-1;caseB-2;caseC,D-3;// 多值合并default-0;};// 甚至可以在块中执行复杂逻辑并使用 yield 返回Stringmessageswitch(type){caseA-{System.out.println(Processing A...);yieldType A Processed;}default-Unknown Type;};3. ⚠️ 特别警示Null 值炸弹Switch 的条件表达式绝不能为 null无论是 String 还是 Enum否则直接抛NullPointerException。务必先判空// ❌ 危险switch(userType){...}// 若 userType 为 null直接崩// ✅ 安全if(userType!null){switch(userType){...}} 第1天 · 核心自查清单知识点 核心口诀☠️ 致命陷阱✅ 最佳实践字面量整 int 小 double长 L 浮 ffloat f 3.14编译错金额用BigDecimalLong 后缀用大写L浮点比较二进制存不准等于号别乱用0.1 0.2 0.3为 false用Math.abs(diff) epsilon或BigDecimalInteger 缓存-128 到 127 是亲戚之外是新欢new Integer(128) new Integer(128)为 false对象比较永远用.equals()空指针拆箱包装类可能是 nullInteger i null; int n i;崩了拆箱前判空或用Optional处理短路逻辑前假后不跑 ()前真后不跑 (ll)误用导致 NPE 或多余计算判空链式调用首选Switch1.7 起能转 String14 箭头更香忘写break导致逻辑穿透Null 值崩盘新版 JDK 优先使用-表达式先判空再 switch 今日作业编写一个工具方法接收两个Double类型的金额参数使用BigDecimal进行加法运算并返回结果需处理 null 值。尝试复现Integer缓存的边界测试127 vs 128并打印它们的hashCode和identityHashCode(System.identityHashCode) 以观察区别。掌握这些细节不仅能帮你通过大厂面试的“基础题”拷打更能避免生产环境中那些隐蔽且难以排查的 Bug。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2452416.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!