第11篇:Java 8 新特性及使用
Java 8 是一次里程碑式的更新,引入了多项革新特性,极大地提升了开辟效率和代码表现力。本文将从 Lambda表达式、Stream API、时间日期API、Optional类 等核心特性出发,联合实战场景和最佳实践,全面剖析Java 8的核心功能。
1. Lambda表达式与函数式编程
1.1 Lambda表达式基础
Lambda表达式答应以简洁的语法实现函数式接口(仅含一个抽象方法的接口),替代传统的匿名内部类。
语法结构:
示例:线程创建对比
- // 传统写法
- new Thread(new Runnable() {
- @Override
- public void run() {
- System.out.println("Thread running");
- }
- }).start();
- // Lambda简化
- new Thread(() -> System.out.println("Thread running")).start();
复制代码 1.2 函数式接口
Java 8 内置了四大核心函数式接口:
- Consumer<T>:接收一个参数,无返回值。
- Consumer<String> print = s -> System.out.println(s);
- print.accept("Hello");
复制代码 - Supplier<T>:无参数,返回一个值。
- Supplier<Double> random = Math::random;
- System.out.println(random.get());
复制代码 - Function<T, R>:接收T范例参数,返回R范例。
- Function<String, Integer> length = String::length;
- System.out.println(length.apply("Java")); // 4
复制代码 - Predicate<T>:接收T范例参数,返回布尔值。
- Predicate<Integer> isEven = n -> n % 2 == 0;
- System.out.println(isEven.test(4)); // true
复制代码 1.3 方法引用
进一步简化Lambda表达式:
- 静态方法引用:ClassName::staticMethod
- List<Integer> numbers = Arrays.asList(1, 2, 3);
- numbers.forEach(System.out::println);
复制代码 - 实例方法引用:instance::method
- String str = "Java";
- Supplier<Integer> length = str::length;
复制代码 - 构造器引用:ClassName::new
- Supplier<List<String>> listSupplier = ArrayList::new;
复制代码 2. Stream API
Stream API 提供了一种声明式处理集合数据的方式,支持次序和并行操纵,极大简化了集合处理。
2.1 Stream操纵分类
- 中心操纵:返回新Stream,延迟执行(如filter, map, sorted)。
- 终端操纵:触发盘算,关闭Stream(如forEach, collect, count)。
示例:集合处理
- List<String> languages = Arrays.asList("Java", "Python", "C++", "Go");
- // 过滤长度>3,转大写,排序,收集为List
- List<String> result = languages.stream()
- .filter(s -> s.length() > 3)
- .map(String::toUpperCase)
- .sorted()
- .collect(Collectors.toList());
- // 输出:[C++, GO, JAVA, PYTHON]
复制代码 2.2 并行流加速处理
通过parallelStream()利用多核CPU提升性能:
- List<Integer> numbers = IntStream.range(1, 10_000_000).boxed().collect(Collectors.toList());
- // 顺序流计算总和
- long sum1 = numbers.stream().mapToLong(Integer::longValue).sum();
- // 并行流计算总和
- long sum2 = numbers.parallelStream().mapToLong(Integer::longValue).sum();
复制代码 2.3 常用收集器(Collectors)
- 转换为集合:toList(), toSet(), toMap()
- 分组统计:groupingBy(), partitioningBy()
- 聚合盘算:summingInt(), averagingDouble(), maxBy()
示例:统计单词频率
- List<String> words = Arrays.asList("apple", "banana", "apple", "orange");
- Map<String, Long> frequency = words.stream()
- .collect(Collectors.groupingBy(Function.identity(), Collectors.counting()));
- // 输出:{apple=2, orange=1, banana=1}
复制代码 3. 时间日期API(java.time)
Java 8 全新的时间日期API解决了旧版Date和Calendar的设计缺陷,提供不可变、线程安全的类。
3.1 核心类
- LocalDate:日期(年代日)。
- LocalDate date = LocalDate.now(); // 当前日期
- LocalDate independenceDay = LocalDate.of(2023, Month.JULY, 4);
复制代码 - LocalTime:时间(时分秒)。
- LocalTime time = LocalTime.parse("15:30:45");
复制代码 - LocalDateTime:日期时间组合。
- LocalDateTime dateTime = LocalDateTime.of(2023, 12, 31, 23, 59);
复制代码 - Duration:时间隔断(基于时间单位)。
- Duration duration = Duration.between(startTime, endTime);
复制代码 - Period:日期隔断(基于年、月、日)。
- Period period = Period.between(startDate, endDate);
复制代码 3.2 格式化与剖析
- DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
- LocalDateTime dateTime = LocalDateTime.parse("2023-08-25 14:30:00", formatter);
- String formatted = dateTime.format(formatter); // "2023-08-25 14:30:00"
复制代码 4. Optional类
Optional用于优雅处理null值,避免空指针异常(NPE)。
4.1 根本使用
- Optional<String> optional = Optional.ofNullable(getName());
- // 存在值时执行操作
- optional.ifPresent(name -> System.out.println("Name: " + name));
- // 提供默认值
- String name = optional.orElse("Unknown");
- // 抛出异常
- String value = optional.orElseThrow(() -> new IllegalArgumentException("值不能为空"));
复制代码 4.2 链式操纵
- Optional<User> user = Optional.ofNullable(getUser());
- String street = user.map(User::getAddress)
- .map(Address::getStreet)
- .orElse("No street");
复制代码 5. 接口默认方法与静态方法
Java 8 答应接口包含默认方法和静态方法,增强接口的扩展能力。
5.1 默认方法
- public interface Vehicle {
- void run();
- default void honk() {
- System.out.println("Beep beep!");
- }
- }
- public class Car implements Vehicle {
- @Override
- public void run() {
- System.out.println("Car is running");
- }
- }
复制代码 5.2 静态方法
- public interface MathUtils {
- static int add(int a, int b) {
- return a + b;
- }
- }
- int sum = MathUtils.add(3, 5); // 8
复制代码 6. 其他新特性
6.1 重复注解
答应在同一位置多次使用相同注解:
- @Retention(RetentionPolicy.RUNTIME)
- public @interface Roles {
- Role[] value();
- }
- @Repeatable(Roles.class)
- public @interface Role {
- String value();
- }
- @Role("admin")
- @Role("user")
- public class User { }
复制代码 6.2 范例注解
注解可应用于任何范例(如泛型、方法返回值):
- List<@NonNull String> list = new ArrayList<>();
复制代码 6.3 方法参数反射
通过Parameter类获取方法参数名(需编译时添加-parameters参数):
- public void printInfo(@NotNull String name, int age) {
- // ...
- }
- Method method = User.class.getMethod("printInfo", String.class, int.class);
- Parameter[] parameters = method.getParameters();
- System.out.println(parameters[0].getName()); // 输出name
复制代码 7. 实战案例与最佳实践
案例1:使用Stream优化集合操纵
- // 传统写法
- List<String> filteredNames = new ArrayList<>();
- for (User user : users) {
- if (user.getAge() > 18) {
- filteredNames.add(user.getName());
- }
- }
- // Stream优化
- List<String> filteredNames = users.stream()
- .filter(user -> user.getAge() > 18)
- .map(User::getName)
- .collect(Collectors.toList());
复制代码 案例2:时间日期盘算
- LocalDate today = LocalDate.now();
- LocalDate nextWeek = today.plusDays(7);
- Period period = Period.between(today, nextWeek);
- System.out.println("间隔天数:" + period.getDays()); // 7
复制代码 最佳实践:
- 优先使用Lambda和Stream:简化代码,提升可读性。
- 避免太过并行化:并行流实用于大数据量且无状态操纵。
- 时间处理使用java.time:替代Date和Calendar。
- 善用Optional减少NPE:但避免滥用(如不要用于集合字段)。
8. 总结
Java 8 的革新特性为开辟者提供了更强盛的工具集:
- Lambda与Stream:函数式编程简化集合操纵。
- 时间日期API:更安全、更直观的时间处理。
- Optional:优雅处理空值,减少NPE。
- 接口增强:默认方法与静态方法提升机动性。
通过合理利用这些特性,可以显着提升代码质量、可维护性和性能。
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |