Java8
一、四大函数式接口
1、Function 函数式接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| package main; import java.sql.SQLOutput; import java.util.function.Function; public class FunctionDemo { public static void main(String[] args) {
Function<String,String> function = (str)->{return str;}; System.out.println(function.apply("dsaf")); } }
|
2、Predicate 断定型接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| package fourinterface; import java.util.function.Predicate;
public class PredicateDemo { public static void main(String[] args) {
Predicate<String> predicate = (str)->{return str.isEmpty();}; System.out.println(predicate.test(" ")); } }
|
3、Consumer 消费性接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| package fourinterface; import java.util.function.Consumer;
public class ConsumerDemo { public static void main(String[] args) {
Consumer<String> consumer = (str)->{ System.out.println(str); }; consumer.accept("hello"); } }
|
4、Supplier 供给型接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| package fourinterface; import java.util.function.Supplier;
public class SupplierDemo { public static void main(String[] args) {
Supplier supplier = ()->{return 1024;}; System.out.println(supplier.get()); } }
|
二、Stream
2.1 预定义数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @Data @AllArgsConstructor public class Student { private String id; private String name; private LocalDate birthday; private int age; private double score; }
final List<Student> students = Lists.newArrayList(); students.add(new Student("1", "张三", LocalDate.of(2009, Month.JANUARY, 1), 12, 12.123)); students.add(new Student("2", "李四", LocalDate.of(2010, Month.FEBRUARY, 2), 11, 22.123)); students.add(new Student("3", "王五", LocalDate.of(2011, Month.MARCH, 3), 10, 32.123));
|
2.2.1 元素数量:counting
1
| students.stream().collect(Collectors.counting())
|
2.2.2 平均值:averagingDouble、averagingInt、averagingLong
1 2 3 4 5 6 7 8
| students.stream().collect(Collectors.averagingDouble(Student::getScore))
students.stream().collect(Collectors.averagingInt(s -> (int)s.getScore())) students.stream().collect(Collectors.averagingLong(s -> (long)s.getScore()))
students.stream().collect(Collectors.averagingInt(Student::getAge)) students.stream().collect(Collectors.averagingDouble(Student::getAge)) students.stream().collect(Collectors.averagingLong(Student::getAge))
|
2.2.3 和:summingDouble、summingInt、summingLong
1 2 3 4 5 6 7
| students.stream().collect(Collectors.summingInt(s -> (int)s.getScore())) students.stream().collect(Collectors.summingDouble(Student::getScore)) students.stream().collect(Collectors.summingLong(s -> (long)s.getScore()))
students.stream().collect(Collectors.summingInt(Student::getAge)) students.stream().collect(Collectors.summingDouble(Student::getAge)) students.stream().collect(Collectors.summingLong(Student::getAge))
|
2.2.4 最大值/最小值元素:maxBy、minBy
1 2 3 4
| students.stream().collect(Collectors.minBy(Comparator.comparing(Student::getAge)))
students.stream().collect(Collectors.maxBy(Comparator.comparing(Student::getAge)))
|
2.2.5 统计结果:summarizingDouble、summarizingInt、summarizingLong
1 2 3 4 5 6 7 8 9 10 11 12 13
| students.stream().collect(Collectors.summarizingInt(s -> (int) s.getScore()))
students.stream().collect(Collectors.summarizingDouble(Student::getScore))
students.stream().collect(Collectors.summarizingLong(s -> (long) s.getScore()))
students.stream().collect(Collectors.summarizingInt(Student::getAge))
students.stream().collect(Collectors.summarizingDouble(Student::getAge))
students.stream().collect(Collectors.summarizingLong(Student::getAge))
|
2.3.1 聚合元素:toList、toSet、toCollection
1 2 3 4 5 6
| final List<String> idList = students.stream().map(Student::getId).collect(Collectors.toList());
final Set<String> idSet = students.stream().map(Student::getId).collect(Collectors.toSet());
final Collection<String> idTreeSet = students.stream().map(Student::getId).collect(Collectors.toCollection(TreeSet::new));
|
2.3.2 聚合元素:toMap、toConcurrentMap
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| final Map<String, Student> map11 = students.stream() .collect(Collectors.toMap(Student::getId, Function.identity())); final Map<String, Student> map2 = students.stream() .collect(Collectors.toMap(Student::getId, Function.identity(), (x, y) -> x));
final Map<String, String> map3 = students.stream() .collect(Collectors.toMap(Student::getId, Student::getName, (x, y) -> x));
final Map<Integer, Student> map5 = students.stream() .collect(Collectors.toMap(Student::getAge, Function.identity(), BinaryOperator.maxBy(Comparator.comparing(Student::getScore))));
|
2.3.3 分组:groupingBy、groupingByConcurrent
1 2 3 4 5 6 7 8 9 10 11 12
| final Map<Integer, List<Student>> map1 = students.stream().collect(Collectors.groupingBy(Student::getAge));
final Map<Integer, Set<Student>> map12 = students.stream().collect(Collectors.groupingBy(Student::getAge, Collectors.toSet()));
final Map<String, Student> map3 = students.stream() .collect(Collectors.groupingBy(Student::getId, Collectors.collectingAndThen(Collectors.toList(), list -> list.get(0))));
final Map<String, Student> map2 = students.stream() .collect(Collectors.toMap(Student::getId, Function.identity(), (x, y) -> x));
|
2.3.4 分组:partitioningBy
1 2 3 4
| final Map<Boolean, List<Student>> map6 = students.stream().collect(Collectors.partitioningBy(s -> s.getAge() > 11));
final Map<Boolean, Set<Student>> map7 = students.stream().collect(Collectors.partitioningBy(s -> s.getAge() > 11, Collectors.toSet()));
|
2.3.5 链接数据:joining
1 2 3 4 5 6
| Stream.of("java", "go", "sql").collect(Collectors.joining());
Stream.of("java", "go", "sql").collect(Collectors.joining(", "));
Stream.of("java", "go", "sql").collect(Collectors.joining(", ", "【", "】"));
|
2.3.6 操作链:collectingAndThen
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| final Map<String, Student> map3 = students.stream() .collect(Collectors.groupingBy(Student::getId, Collectors.collectingAndThen(Collectors.toList(), list -> list.get(0))));
students.stream() .collect( Collectors.collectingAndThen(Collectors.toList(), ( list -> list.stream() .filter(s -> (LocalDate.now().getYear() - s.getBirthday().getYear()) != s.getAge()) .collect(Collectors.toList())) ) );
students.stream() .filter(s -> (LocalDate.now().getYear() - s.getBirthday().getYear()) != s.getAge()) .collect(Collectors.toList());
|
2.3.7 操作后聚合:mapping
1 2 3 4 5 6 7 8
| students.stream() .collect(Collectors.mapping(Student::getName, Collectors.toList()));
students.stream() .map(Student::getName) .collect(Collectors.toList());
|
2.3.8 聚合后操作:reducing
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| students.stream() .map(Student::getScore) .collect(Collectors.reducing(Double::sum));
students.stream() .map(Student::getScore) .collect(Collectors.reducing(0.0, Double::sum));
students.stream() .collect(Collectors.reducing(0.0, Student::getScore, Double::sum));
students.stream().map(Student::getScore).reduce(Double::sum);
students.stream().map(Student::getScore).reduce(0.0, Double::sum);
|
三、Optional
主要作用是消除 NullPointException
::: code-tabs#java
@tab Java7
1 2
| List<String> list = getList(); List<String> listOpt = list != null ? list : new ArrayList<>();
|
@tab Java8
1 2
| List<String> listOpt = Optional.ofNullable(getList()) .orElse(new ArrayList<>());
|
:::
举例:
假设,我们有一个User类,内部有个Address类,在内部有个street属性,我们现在想要获取一个User对象的street值。如果是以前,我们需要各种判断是否是null,代码会写成这样:
::: code-tabs#java
@tab Java7
1 2 3 4 5 6 7 8 9 10 11
| User user = getUser(); if (user != null) { Address address = user.getAddress(); if (address != null) { String street = address.getStreet(); if (street != null) { return street; } } } return "not specified";
|
@tab Java8
1 2 3 4
| String result = Optional.ofNullable(getUser()) .map(User::getAddress) .map(Address::getStreet) .orElse("not specified");
|
:::
四、Time
在旧版的 Java 中,日期时间 API 存在诸多问题:
Java 8 在 java.time 包下提供了很多新的 API。
4.1 LocalDateTime
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| LocalDateTime currentTime = LocalDateTime.now(); System.out.println("当前时间: " + currentTime); LocalDate date1 = currentTime.toLocalDate(); System.out.println("date1: " + date1); Month month = currentTime.getMonth(); int day = currentTime.getDayOfMonth(); int seconds = currentTime.getSecond(); System.out.println("月: " + month +", 日: " + day +", 秒: " seconds); LocalDateTime date2 = currentTime.withDayOfMonth(10).withYe(2012); System.out.println("date2: " + date2);
LocalDate date3 = LocalDate.of(2014, Month.DECEMBER, 12); System.out.println("date3: " + date3);
LocalTime date4 = LocalTime.of(22, 15); System.out.println("date4: " + date4);
LocalTime date5 = LocalTime.parse("20:15:30"); System.out.println("date5: " + date5);
|
输出:
1 2 3 4 5 6 7
| 当前时间: 2016-04-15T16:55:48.668 date1: 2016-04-15 月: APRIL, 日: 15, 秒: 48 date2: 2012-04-10T16:55:48.668 date3: 2014-12-12 date4: 22:15 date5: 20:15:30
|
4.2 ZonedDateTime
1 2 3 4 5 6 7 8 9
| ZonedDateTime date1 = ZonedDateTime.parse("2015-12-03T10:15:30+05:30[Asia/Shanghai]"); System.out.println("date1: " + date1); ZoneId id = ZoneId.of("Europe/Paris"); System.out.println("ZoneId: " + id); ZoneId currentZone = ZoneId.systemDefault(); System.out.println("当期时区: " + currentZone);
|
输出:
1 2 3
| date1: 2015-12-03T10:15:30+08:00[Asia/Shanghai] ZoneId: Europe/Paris 当期时区: Asia/Shanghai
|
[参考]https://xie.infoq.cn/article/df361a1280773d32208550293