如果你也是从public static void main(String[] args)和System.out.println()开始Java生涯的,那也是Java老油条了。在日常的业务开发中,我们每天都在写着增删改查的逻辑,有时候会觉得Java有点笨重,语法有点啰嗦。
但其实,Java在不断进化。从我们熟悉的Java 8到现在的Java 25,它多了很多实用的新特性和一些不为人知的老技巧。用好它们,不仅能让代码更简洁,还能在同事面前小小地秀一下。
下面就聊几个我自己在工作中觉得特别好用的技巧,看看你用过几个。
用枚举(Enum)干点正事
很多人对枚举的印象还停留在定义一组常量,比如MALE,FEMALE。但其实,枚举远比这强大,它甚至可以拥有自己的方法和实现,非常适合用来替代一堆if-else或者switch的策略逻辑。
举个例子,假设我们有不同的会员等级,需要计算折扣后的价格:
java
体验AI代码助手
代码解读
复制代码
// 利用枚举实现不同会员的折扣策略 public enum MemberType { REGULAR { @Override public double applyDiscount(double price) { return price * 0.98; // 普通会员98折 } }, VIP { @Override public double applyDiscount(double price) { return price * 0.9; // VIP会员9折 } }, PREMIUM { @Override public double applyDiscount(double price) { return price * 0.8; // 高级会员8折 } }; public abstract double applyDiscount(double price); } // 使用起来非常清晰 public class PriceCalculator { public static void main(String[] args) { double originalPrice = 100.0; double vipPrice = MemberType.VIP.applyDiscount(originalPrice); System.out.println("VIP会员价:" + vipPrice); // 输出:VIP会员价:90.0 } }
这样写,每种会员的折扣逻辑都封装在自己的枚举实例里,代码清晰,扩展起来也方便。以后要加新的会员等级,只需要添加一个新的枚举实例就行,完全符合开闭原则。
用记录(Record)告别样板代码
自从Java 16引入了Record类型,我写DTO(数据传输对象)的幸福感直线上升。以前为了定义一个简单的数据载体,得手动写一堆的getter,setter,equals(),hashCode()和toString(),或者依赖Lombok。
现在,一行代码就够了。
java
体验AI代码助手
代码解读
复制代码
// 以前的写法 // public class User { // private final String username; // private final String email; // // ... 一大堆 getter, equals, hashCode, toString ... // } // 现在用Record public record UserProfile(String username, String email) {} // 使用 public class Main { public static void main(String[] args) { UserProfile user = new UserProfile("dev_user", "user@example.com"); System.out.println(user.username()); // 直接调用,像方法一样 System.out.println(user); // 自带了很好的toString()实现 } }
编译器会自动帮你生成所有必需的方法,代码瞬间清爽了很多。
类型安全的ID,防止传错参数
在业务代码里,经常会用Long或者String来表示各种ID,比如userId,orderId,productId。这样做的风险是,方法的参数很容易传混。
java
体验AI代码助手
代码解读
复制代码
// 很容易写错的调用 public void processOrder(Long userId, Long orderId) { // ... } // 调用时可能不小心把两个ID搞反 processOrder(orderId, userId); // 编译器不会报错,但逻辑全错了
那就可以利用Record(或者普通类) 来给ID一层包装,增加类型安全性,让编译器在编码阶段就帮我们发现错误。
java
体验AI代码助手
代码解读
复制代码
public record UserId(long value) {} public record OrderId(long value) {} public class OrderService { public void processOrder(UserId userId, OrderId orderId) { System.out.println("处理用户 " + userId.value() + " 的订单 " + orderId.value()); } public static void main(String[] args) { OrderService service = new OrderService(); UserId userId = new UserId(1001L); OrderId orderId = new OrderId(9527L); service.processOrder(userId, orderId); // 正确 // service.processOrder(orderId, userId); // 这行代码会直接编译失败,安全! } }
这个小改动能避免很多难以排查的线上问题。
用 Stream API 告别 for 循环
从Java 8开始,Stream API 就已经是处理集合的标配了。如果还在用for循环和一堆if来做筛选和转换,那代码读起来会很费劲。Stream API可以用一种更声明式、更流畅的方式来操作数据。
比如,我们要从一堆产品里,筛选出价格大于500的,然后取出它们的名字:
java
体验AI代码助手
代码解读
复制代码
import java.util.List; import java.util.stream.Collectors; public class StreamDemo { record Product(String name, double price) {} public static void main(String[] args) { List<Product> products = List.of( new Product("笔记本电脑", 5999.0), new Product("鼠标", 299.0), new Product("机械键盘", 799.0) ); // 使用Stream API List<String> expensiveProductNames = products.stream() .filter(p -> p.price() > 500.0) // 筛选价格 .map(Product::name) // 提取名称 .collect(Collectors.toList()); // 收集成列表 System.out.println(expensiveProductNames); // 输出: [笔记本电脑, 机械键盘] } }
链式调用,一气呵成,代码的意图一目了然。
用文本块(Text Blocks)优雅地写多行字符串
以前在Java代码里拼接SQL或者JSON字符串,简直是一场灾难,充满了+号和\n转义符,可读性极差。Java 15引入的文本块(Text Blocks)彻底解决了这个问题。
看看对比:
java
体验AI代码助手
代码解读
复制代码
public class TextBlockDemo { public static void main(String[] args) { // 以前的方式,又丑又容易出错 String oldJson = "{\n" + " "name": "Alice",\n" + " "age": 30\n" + "}"; // 现在用文本块,所见即所得 String newJson = """ { "name": "Alice", "age": 30 } """; System.out.println(oldJson.equals(newJson)); // 输出: true } }
用三个双引号"""包起来,字符串的格式就能完全保留,代码干净多了。
还在用!= null?试试Optional
空指针异常(NullPointerException)应该不少 Java 开发者都很熟了。为了避免它,我们代码里堆满了if (obj != null)的判断。Optional的出现就是为了更优雅地处理可能为空的情况。
java
体验AI代码助手
代码解读
复制代码
import java.util.Optional; public class UserRepository { // 模拟从数据库查找用户 public Optional<String> findUserNameById(long id) { if (id == 1L) { return Optional.of("Alice"); } return Optional.empty(); // 表示没找到 } public static void main(String[] args) { UserRepository repo = new UserRepository(); // 优雅地处理 repo.findUserNameById(1L) .ifPresentOrElse( name -> System.out.println("找到用户:" + name), () -> System.out.println("用户不存在") ); // 获取值,如果不存在则提供一个默认值 String userName = repo.findUserNameById(2L).orElse("默认用户"); System.out.println("查询ID为2的用户:" + userName); } }
用Optional不仅能让代码意图更明确(这个方法的返回值可能为空),还能通过链式调用写出更流畅的代码。
管理好Java环境,才能玩转新特性
看到这里,你可能已经注意到,上面提到的不少特性都和Java版本有关。比如Record是Java 16的,文本块是Java 15的,ifPresentOrElse是Java 9的。在实际工作中,我们经常会遇到这样的情况:
老项目A还在用稳定的Java 8。
新项目B想尝试Java 17的LTS版本。
自己想学习一下最新的Java 25。
在电脑上同时管理这么多Java版本,配置环境变量,为不同项目切换JDK,挺让人破防的。每次切换环境,都要敲一堆命令,或者在IDE里改来改去,很影响开发效率。
所以,这时候就需要ServBay。
它本来是一个集成了PHP、Node.js等环境的Web开发工具,但我发现它对Java的支持也做得特别好。最让我喜欢的一点是,它可以一键安装和管理多个Java版本。我可以同时安装Java 8、11、17、21等多个版本,它们之间完全隔离,互不干扰。
而且,我还可以为不同的项目指定使用不同的Java版本。比如,可以设置项目A的运行环境就是Java 8,项目B就是Java 23。这样一来,在切换项目时,根本不用关心JDK版本的问题,ServBay都帮你自动处理好了。
对于我们Java开发者来说,这意味着可以把更多精力放在代码和技术本身,而不是被环境配置这些琐事消耗。
总结
Java依然是一门生命力旺盛的语言。学习和使用这些技巧,可以让我们的代码质量更高,开发体验也更好。而借助像ServBay这样的工具,又能帮我们轻松搞定复杂的环境管理问题,让我们能更专注于写出优秀的代码。
你还有什么私藏的Java技巧吗?欢迎在评论区分享交流。