Java 8 新特性——实践篇

参考

Java8新特性

重要更新:Lambda 表达式和Stream API

Lambda 表达式

Lambda 表达式引入之前:

举个场景例子:当我们要对一个班级里的学生对象里各种成绩进行过滤时,比如大于85分获得A的学生集合,最初的方式是写不同的方法处理不同的科目成绩过滤;再后面就可以用策略模式,声明一个接口ScoreFilterStrategy,针对不同的科目实现不同的策略算法。再优化一下,我们可以对策略模式进行升级,直接用匿名内部类实现我们的接口ScoreFilterStrategy,自定义策略实现。但基于其代码的繁琐性,我们可以使用Lambda 表达式进行函数式编程优化,更可以对集合进行Stream API流的调用处理来实现想要的效果。参考如下:

Lambda 是一个匿名函数,我们可以把Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。

Lambda 表达式语法

Lambda 表达式在Java 语言中引入了一个新的语法元素和操作符。这个操作符为“->” ,该操作符被称为Lambda 操作符或剪头操作符。它将Lambda 分为两个部分:

左侧:指定了Lambda 表达式需要的所有参数

右侧:指定了Lambda 体,即Lambda 表达式要执行的功能

Lambda 表达式需要函数式接口的支持。

示例:

  1. public class LambdaTest {
  2. //语法格式一:无参,无返回值,Lambda 体只需一条语句
  3. @Test
  4. public void test1() {
  5. Runnable r = () -> System.out.println("helo world");
  6. r.run();
  7. }
  8. //语法格式二:Lambda 需要一个参数
  9. @Test
  10. public void test2() {
  11. Consumer<String> consumer = (str) -> System.out.println(str);
  12. consumer.accept("I am Batman");
  13. }
  14. //语法格式三:Lambda 只需要一个参数时,参数的小括号可以省略
  15. @Test
  16. public void test3() {
  17. Consumer<String> consumer = str -> System.out.println(str);
  18. consumer.accept("I am Batman");
  19. }
  20. //语法格式四:Lambda 需要两个参数,并且有返回值,并且lambda体中有多条语句时要加{}
  21. @Test
  22. public void test4() {
  23. Comparator<Integer> comparator = (x, y) -> {
  24. System.out.println("比较数据");
  25. return Integer.compare(x, y);
  26. };
  27. comparator.compare(1, 2);
  28. }
  29. //语法格式五:当Lambda 体只有一条语句时,return 与大括号可以省略
  30. @Test
  31. public void test5() {
  32. Comparator<Integer> comparator = (x, y) -> Integer.compare(x, y);
  33. comparator.compare(1, 2);
  34. }
  35. //语法格式六:Lambda 表达式的参数列表上的参数类型可以省略不写,因为JVM编译器(javac)会通过程序上下文推动出数据类型,
  36. // 即类型推断。这是JDK1.8的新特性,在JDK1.7上写类型推断的代码是编译不通过的。
  37. @Test
  38. public void test6() {
  39. Comparator<Integer> comparator = (x, y) -> Integer.compare(x, y);
  40. comparator.compare(1, 2);
  41. }
  42. //类型推断实例
  43. @Test
  44. public void test7() {
  45. String[] strings = {"a", "b", "ddd"};
  46. //如果像下面这样写就无法通过上下文进行类型推断
  47. // String[] string2;
  48. // string2 = {"a", "b", "ddd"};
  49. //后面的new ArrayList<>();尖括号里面也可以省略不写也是通过类型推断得来的。
  50. List<String> list = new ArrayList<>();
  51. //创建的new HashMap<>()尖括号里面也可以省略不写也是通过类型推断得来的,他会采用operate方法里的泛型格式
  52. operate(new HashMap<>());
  53. }
  54. private void operate(Map<String,String> map){
  55. }
  56. }

函数式接口

接口中只包含一个抽象方法的接口,称为函数式接口。我们可以使用@FunctionalInterface注解修饰该接口,用于检测是否是函数式接口。

示例:

  1. @FunctionalInterfacepublic interface MyFunc { Integer getValue(Integer t);}
  1. @FunctionalInterfacepublic interface CustomFunc { String execute(String str);}
  1. @FunctionalInterface
  2. public interface CauclateFunc<T,R> {
  3. R getValue(T t1,T t2);
  4. }
  1. public class LambdaTest2 {
  2. @Test
  3. public void test() {
  4. System.out.println(operate(100, x -> x * x));
  5. }
  6. //函数式接口作为方法参数传递,这样我们便能在调用方法时自定义我们的lambda函数的操作
  7. private Integer operate(Integer t, MyFunc myFunc) {
  8. return myFunc.getValue(t);
  9. }
  10. @Test
  11. public void test2() {
  12. System.out.println(concate("hello",str -> str + " world"));
  13. String concate = concate("I", str -> {
  14. System.out.println("prepare...");
  15. return str + " am " + "batman";
  16. });
  17. System.out.println(concate);
  18. }
  19. private String concate(String str,CustomFunc customFunc){
  20. return customFunc.execute(str);
  21. }
  22. @Test
  23. public void test3() {
  24. Long calcute = calcute(100L, 200L, (x, y) -> x * y);
  25. System.out.println(calcute);
  26. Long calcute2 = calcute(100L, 200L, (x, y) -> x - y);
  27. System.out.println(calcute2);
  28. }
  29. private Long calcute(Long l1, Long l2, CauclateFunc<Long, Long> cauclateFunc) {
  30. return cauclateFunc.getValue(l1, l2);
  31. }
  32. }
  1. 输出:
  2. hello world
  3. prepare...
  4. I am batman
  5. 20000
  6. -100

Java 内置四大核心函数式接口

其他子接口是在四大核心函数式接口的基础上进行个性化的参数添加和处理定义:

四大核心函数式接口使用示例:

  1. public class FunctionalInterfaceTest {
  2. /**
  3. * Consumer<T> :消费型接口:接收一个参数进行消费处理,无返回
  4. * void accept(T t)
  5. */
  6. @Test
  7. public void test() {
  8. call("麻包锅", name -> System.out.println(name + "被老师点名了!"));
  9. }
  10. //麻包锅被老师点名了!
  11. private void call(String name, Consumer<String> consumer) {
  12. consumer.accept(name);
  13. }
  14. /**
  15. * Supplier<T> :供给型接口 :返回一些东西
  16. * T get();
  17. */
  18. @Test
  19. public void test2() {
  20. //注意加括号,int强转是针对整个只而不是Math.random()返回的,否则为0
  21. List<Integer> list = supply(5, () -> (int) (Math.random() * 100));
  22. System.out.println(list);
  23. }
  24. //[78, 96, 66, 20, 0]
  25. private List<Integer> supply(Integer num, Supplier<Integer> supplier) {
  26. List<Integer> list = new ArrayList<>();
  27. for (int i = 0; i < num; i++) {
  28. list.add(supplier.get());
  29. }
  30. return list;
  31. }
  32. /**
  33. * Function<T, R>:函数型接口 :对T类型参数对象处理返回R类型对象
  34. * R apply(T t);
  35. */
  36. @Test
  37. public void test3() {
  38. String concate = concate("希望耗子尾汁!", str -> "老同志," + str);
  39. System.out.println(concate);
  40. String filterStr = filterStr("偷袭, 骗我老同志! ", s -> s.replace(" ",""));
  41. System.out.println(filterStr);
  42. }
  43. //老同志,希望耗子尾汁!
  44. //偷袭,骗我老同志!
  45. private String concate(String str, Function<String,String> function) {
  46. return function.apply(str);
  47. }
  48. private String filterStr(String str, Function<String,String> function) {
  49. return function.apply(str);
  50. }
  51. /**
  52. * Predicate<T> :断言型接口
  53. * boolean test(T t);
  54. */
  55. @Test
  56. public void test4() {
  57. List<Integer> numList = new ArrayList<>();
  58. numList.add(1);
  59. numList.add(13);
  60. numList.add(23);
  61. numList.add(67);
  62. List<Integer> list = check(numList, n -> n > 15);
  63. System.out.println(list);
  64. }
  65. //[23, 67]
  66. List<Integer> check(List<Integer> list,Predicate<Integer> predicate){
  67. List<Integer> newList = new ArrayList<>();
  68. for (Integer num : list) {
  69. if (predicate.test(num)){
  70. newList.add(num);
  71. }
  72. }
  73. return newList;
  74. }
  75. }

方法引用与构造器引用

方法引用

当要传递给Lambda体的操作,如果Lambda 体中的内容有方法实现了,就可以使用“方法引用”。即方法引用是lambda表达式的另一种表现形式。

有以下三种语法形式:

  • 对象::实例方法
  • 类::静态方法
  • 类::实例方法

注意:

  • lambda体中调用方法的参数列表和返回值类型,要与函数式接口中的抽象方法的参数列表和返回值类型保持一致。
  • 如果lambda参数列表中的第一个参数是示例方法的调用者,第二个参数是示例方法的参数时,可以使用ClassName::methodName。

构造器引用

格式:ClassName::new

构造器引用与函数式接口相结合,自动与函数式接口中方法兼容。

注意:

  • 当我们把构造器引用赋值给定义的方法时,需要调用的构造器的参数列表要与函数式接口中抽象方法的参数列表一致!

数组引用

格式:type[] :: new

示例:

  1. public class MethodRefTest {
  2. /**
  3. * 对象::实例方法
  4. * 注意方法引用使用时:lamda表达式需要实现的抽象方法的参数类型和返回值要与当前调用的方法的参数类型和返回值保持一致
  5. * Consumer<T> : void accept(T t);
  6. * PrintStream : void println(String x);
  7. * 当调用方法有一个参数无返回值时适用于Consumer的消费型接口
  8. */
  9. @Test
  10. public void test() {
  11. Consumer<String> consumer = System.out::println;
  12. consumer.accept("朋友们好啊!");
  13. }
  14. /**
  15. * Supplier<T> T get();
  16. * 调用方法无参数有返回值时适用于Supplier的供给型接口
  17. * 典型的如对象的get方法
  18. */
  19. @Test
  20. public void test2() {
  21. User user = new User();
  22. Supplier<String> supplier = user::getName;
  23. System.out.println(supplier.get());
  24. }
  25. /**
  26. * 类::静态方法
  27. * Comparator<T> : int compare(T o1, T o2);
  28. * Integer : static int compare(int x, int y)
  29. */
  30. @Test
  31. public void test3() {
  32. List<Integer> numList = new ArrayList<>();
  33. numList.add(111);
  34. numList.add(13);
  35. numList.add(2);
  36. numList.add(67);
  37. // Comparator<Integer> comparator = Integer::compare;
  38. // Collections.sort(numList,comparator);
  39. Collections.sort(numList, Integer::compare);
  40. System.out.println(numList);
  41. }
  42. //[2, 13, 67, 111]
  43. /**
  44. * 类::实例方法
  45. * 使用条件:如果lambda参数列表中的第一个参数是示例方法的调用者,第二个参数是示例方法的参数时
  46. * BiPredicate<T, U> : boolean test(T t, U u);
  47. * String : boolean equals(Object anObject)
  48. */
  49. @Test
  50. public void test4() {
  51. boolean isEqual = checkEqual("music", "Music", String::equals);
  52. System.out.println(isEqual);
  53. }
  54. //false
  55. private boolean checkEqual(String a, String b, BiPredicate<String, String> biPredicate) {
  56. return biPredicate.test(a, b);
  57. }
  58. /**
  59. * 构造器引用
  60. * 需要调用的构造器的参数列表要与函数式接口中抽象方法的参数列表一致
  61. * Supplier<T> : T get(); 对于无参构造器
  62. * Function<T, R> : R apply(T t); 对于一个参数构造器
  63. * BiFunction<T, U, R> : R apply(T t, U u); 对于两个参数构造器
  64. */
  65. @Test
  66. public void test5() {
  67. //函数式编程写法
  68. Supplier<User> supplier1 = () -> new User();
  69. //构造器引用
  70. Supplier<User> supplier = User::new;
  71. //一个参数构造器
  72. Function<String, User> function = User::new;
  73. User user = function.apply("年轻人");
  74. System.out.println(user);//User{id=0, age=0, name='年轻人'}
  75. BiFunction<Integer, String, User> biFunction = User::new;
  76. User user1 = biFunction.apply(69, "老同志");
  77. System.out.println(user1);//User{id=0, age=69, name='老同志'}
  78. }
  79. /**
  80. * 数组引用
  81. */
  82. @Test
  83. public void test6() {
  84. Function<Integer, String[]> function1 = x -> new String[x];
  85. Function<Integer, String[]> function = String[]::new;
  86. String[] strings = function.apply(5);
  87. System.out.println(strings.length);//5
  88. }
  89. }
  1. public class User {
  2. private int id;
  3. private int age;
  4. private String name;
  5. public User() {
  6. }
  7. public User(int age, String name) {
  8. this.age = age;
  9. this.name = name;
  10. }
  11. public User(String name) {
  12. this.name = name;
  13. }
  14. public int getId() {
  15. return id;
  16. }
  17. public void setId(int id) {
  18. this.id = id;
  19. }
  20. public int getAge() {
  21. return age;
  22. }
  23. public void setAge(int age) {
  24. this.age = age;
  25. }
  26. public String getName() {
  27. return name;
  28. }
  29. public void setName(String name) {
  30. this.name = name;
  31. }
  32. @Override
  33. public String toString() {
  34. return "User{" +
  35. "id=" + id +
  36. ", age=" + age +
  37. ", name='" + name + '\'' +
  38. '}';
  39. }
  40. }

Stream API

Stream 是Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。使用Stream API 对集合数据进行操作,就类似于使用SQL 执行的数据库查询。也可以使用Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式

集合讲的是数据,Stream流讲的是计算。

注意:

  • Stream 自己不会存储元素
  • Stream 不会改变源对象。相反,他们会返回一个持有结果的新Stream。
  • Stream 操作是延迟执行的,这意味着他们会等到需要结果的时候才执行。

Stream 的操作有三个步骤:

  • 创建Stream

    一个数据源(如:集合、数组),获取一个流
  • 中间操作

    一个中间操作链,对数据源的数据进行处理
  • 终止操作

    一个终止操作,执行中间操作链,并产生结果

创建Stream

创建Stream流的4种方式示例:

  1. public class StreamTest {
  2. //创建Stream流的4种方式
  3. @Test
  4. public void test() {
  5. //1、集合创建流:通过Collection系列集合提供的stream()或parallelStream()方法创建流
  6. List<String> list = new ArrayList<>();
  7. //串行流
  8. Stream<String> stream = list.stream();
  9. //并行流
  10. Stream<String> parallelStream = list.parallelStream();
  11. //2、数组创建流:通过Arrays.stream(T[] array)静态方法获取数组流
  12. User[] users = new User[5];
  13. Stream<User> userStream = Arrays.stream(users);
  14. //3、Stream创建流: 通过Stream.of(T... values)静态方法创建流
  15. Stream<Integer> integerStream = Stream.of(1, 2, 3, 66, 75);
  16. //4、创建无限流
  17. // A:通过Stream.iterate(final T seed, final UnaryOperator<T> f)传入seed和一元函数式接口实现无限流
  18. Stream<Integer> stream1 = Stream.iterate(1, x -> x + 1);
  19. stream1.limit(5).forEach(System.out::println);
  20. // B:通过Stream.igenerate(Supplier<T> s) 提供一个供给型接口实现无限流
  21. Stream<Double> stream2 = Stream.generate(()->Math.random());
  22. stream2.limit(5).forEach(System.out::println);
  23. }
  24. }

输出:

  1. 1
  2. 2
  3. 3
  4. 4
  5. 5
  6. 0.19737667335799347
  7. 0.4379300542517345
  8. 0.626269580987739
  9. 0.8557261379085842
  10. 0.09320455087266999

中间操作

多个中间操作可以连接起来形成一个流水线,除非流水线上触发终止操作,否则中间操作不会执行任何的处理

而在终止操作时一次性全部处理,称为“惰性求值”或者叫延迟加载

中间操作示例:

  1. public class StreamTest2 {
  2. List<User> users = Arrays.asList(new User(1, 27, "大青山", Arrays.asList("shanghai", "beijing")),
  3. new User(2, 47, "池寒枫", Arrays.asList("shanghai", "beijing")),
  4. new User(3, 25, "艾米", Arrays.asList("shanghai", "beijing")),
  5. new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  6. new User(5, 380, "霍恩斯", Arrays.asList("shanghai", "beijing")),
  7. new User(5, 500, "霍恩斯", Arrays.asList("shanghai", "beijing")),
  8. new User(5, 320, "霍恩斯1", Arrays.asList("shanghai", "beijing")),
  9. new User(5, 600, "霍恩斯2", Arrays.asList("guangzhou", "shengzhen")));
  10. //中间操作
  11. @Test
  12. public void test() {
  13. //filter: boolean test(T t); 通过断言型接口参数过滤
  14. //forEach 是内部迭代,由Stream api 完成,相对的外部迭代就是我们自己写迭代代码iterator
  15. users.stream().filter(user -> user.getAge() > 23).forEach(System.out::println);
  16. }
  17. @Test
  18. public void test2() {
  19. //limit 截断流,截取maxSize个,当截取到足够的个数后便会短路,不再迭代下去
  20. users.stream().filter(user -> user.getAge() > 23)
  21. .limit(2).forEach(System.out::println);
  22. }
  23. /**
  24. * User{id=1, age=27, name='大青山'}
  25. * User{id=2, age=47, name='池寒枫'}
  26. */
  27. @Test
  28. public void test3() {
  29. //skip 跳过n个
  30. //distinct 去重 如果是对象需要重写hashCode和equal方法
  31. users.stream().filter(user -> user.getAge() > 23)
  32. .skip(2).distinct().forEach(System.out::println);
  33. }
  34. /**
  35. * User{id=3, age=25, name='艾米'}
  36. * User{id=5, age=500, name='霍恩斯'}
  37. */
  38. @Test
  39. public void test4() {
  40. //map 映射 参数是函数式接口Function<? super T, ? extends R>
  41. users.stream().map(user -> user.getName()).forEach(System.out::println);
  42. }
  43. /**
  44. * 大青山
  45. * 池寒枫
  46. * 艾米
  47. * 池傲天
  48. * 霍恩斯
  49. * 霍恩斯
  50. */
  51. @Test
  52. public void test5() {
  53. //flatMap 映射成流,将流中的每个值都映射成流,把所有的流连接成一个流。
  54. // 参数是函数式接口Function<? super T, ? extends Stream<? extends R>> mapper ,注意出参必须是个Stream流对象。因此lambda表达式的返回值要是个流
  55. Stream<String> stream = users.stream().flatMap(user -> user.getAddress().stream());
  56. // List<String> collect = stream.collect(Collectors.toList());
  57. // System.out.println(collect);
  58. stream.forEach(System.out::println);
  59. }
  60. @Test
  61. public void test6() {
  62. //sorted() 排序:沒有传参数默认自然排序: Comparable :注意,要排序的对象必须实现Comparable才能调用自然排序sorted()
  63. /**像String就能调用sorted() 排序,因为他实现了Comparable。
  64. * 否则会报错:java.lang.ClassCastException: com.self.practice.lambda.User cannot be cast to java.lang.Comparable
  65. * public final class String
  66. * implements java.io.Serializable, Comparable<String>, CharSequence
  67. */
  68. //sorted(Comparator<? super T> comparator); 定制排序 Comparator
  69. Stream<User> stream = users.stream().sorted((e1, e2) -> {
  70. if (e1.getName().equals(e2.getName())) {
  71. return e1.getAge().compareTo(e2.getAge());//倒序取反就行,加个负号 -
  72. } else {
  73. return e1.getName().compareTo(e2.getName());
  74. }
  75. });
  76. stream.forEach(System.out::println);
  77. }
  78. }
  79. /**
  80. * User{id=1, age=27, name='大青山'}
  81. * User{id=4, age=18, name='池傲天'}
  82. * User{id=2, age=47, name='池寒枫'}
  83. * User{id=3, age=25, name='艾米'}
  84. * User{id=5, age=380, name='霍恩斯'}
  85. * User{id=5, age=500, name='霍恩斯'}
  86. * User{id=5, age=320, name='霍恩斯1'}
  87. * User{id=5, age=600, name='霍恩斯2'}
  88. */

终止操作





Collector 接口中方法的实现决定了如何对流执行收集操作(如收集到List、Set、Map)。但是Collectors 实用类提供了很多静态方法,可以方便地创建常见收集器实例。

示例:

  1. public class StreamTest {
  2. //创建Stream流的4种方式
  3. @Test
  4. public void test() {
  5. //1、集合创建流:通过Collection系列集合提供的stream()或parallelStream()方法创建流
  6. List<String> list = new ArrayList<>();
  7. //串行流
  8. Stream<String> stream = list.stream();
  9. //并行流
  10. Stream<String> parallelStream = list.parallelStream();
  11. //2、数组创建流:通过Arrays.stream(T[] array)静态方法获取数组流
  12. User[] users = new User[5];
  13. Stream<User> userStream = Arrays.stream(users);
  14. //3、Stream创建流: 通过Stream.of(T... values)静态方法创建流
  15. Stream<Integer> integerStream = Stream.of(1, 2, 3, 66, 75);
  16. //4、创建无限流
  17. // A:通过Stream.iterate(final T seed, final UnaryOperator<T> f)传入seed和一元函数式接口实现无限流
  18. Stream<Integer> stream1 = Stream.iterate(1, x -> x + 1);
  19. stream1.limit(5).forEach(System.out::println);
  20. // B:通过Stream.igenerate(Supplier<T> s) 提供一个供给型接口实现无限流
  21. Stream<Double> stream2 = Stream.generate(()->Math.random());
  22. stream2.limit(5).forEach(System.out::println);
  23. }
  24. }
  1. public class StreamTest2 {
  2. List<User> users = Arrays.asList(new User(1, 27, "大青山", Arrays.asList("shanghai", "beijing")),
  3. new User(2, 47, "池寒枫", Arrays.asList("shanghai", "beijing")),
  4. new User(3, 25, "艾米", Arrays.asList("shanghai", "beijing")),
  5. new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  6. new User(5, 380, "霍恩斯", Arrays.asList("shanghai", "beijing")),
  7. new User(5, 500, "霍恩斯", Arrays.asList("shanghai", "beijing")),
  8. new User(5, 320, "霍恩斯1", Arrays.asList("shanghai", "beijing")),
  9. new User(5, 600, "霍恩斯2", Arrays.asList("guangzhou", "shengzhen")));
  10. //中间操作
  11. @Test
  12. public void test() {
  13. //filter: boolean test(T t); 通过断言型接口参数过滤
  14. //forEach 是内部迭代,由Stream api 完成,相对的外部迭代就是我们自己写迭代代码iterator
  15. users.stream().filter(user -> user.getAge() > 23).forEach(System.out::println);
  16. }
  17. @Test
  18. public void test2() {
  19. //limit 截断流,截取maxSize个,当截取到足够的个数后便会短路,不再迭代下去
  20. users.stream().filter(user -> user.getAge() > 23)
  21. .limit(2).forEach(System.out::println);
  22. }
  23. /**
  24. * User{id=1, age=27, name='大青山'}
  25. * User{id=2, age=47, name='池寒枫'}
  26. */
  27. @Test
  28. public void test3() {
  29. //skip 跳过n个
  30. //distinct 去重 如果是对象需要重写hashCode和equal方法
  31. users.stream().filter(user -> user.getAge() > 23)
  32. .skip(2).distinct().forEach(System.out::println);
  33. }
  34. /**
  35. * User{id=3, age=25, name='艾米'}
  36. * User{id=5, age=500, name='霍恩斯'}
  37. */
  38. @Test
  39. public void test4() {
  40. //map 映射 参数是函数式接口Function<? super T, ? extends R>
  41. users.stream().map(user -> user.getName()).forEach(System.out::println);
  42. }
  43. /**
  44. * 大青山
  45. * 池寒枫
  46. * 艾米
  47. * 池傲天
  48. * 霍恩斯
  49. * 霍恩斯
  50. */
  51. @Test
  52. public void test5() {
  53. //flatMap 映射成流,将流中的每个值都映射成流,把所有的流连接成一个流。
  54. // 参数是函数式接口Function<? super T, ? extends Stream<? extends R>> mapper ,注意出参必须是个Stream流对象。因此lambda表达式的返回值要是个流
  55. Stream<String> stream = users.stream().flatMap(user -> user.getAddress().stream());
  56. // List<String> collect = stream.collect(Collectors.toList());
  57. // System.out.println(collect);
  58. stream.forEach(System.out::println);
  59. }
  60. @Test
  61. public void test6() {
  62. //sorted() 排序:沒有传参数默认自然排序: Comparable :注意,要排序的对象必须实现Comparable才能调用自然排序sorted()
  63. /**像String就能调用sorted() 排序,因为他实现了Comparable。
  64. * 否则会报错:java.lang.ClassCastException: com.self.practice.lambda.User cannot be cast to java.lang.Comparable
  65. * public final class String
  66. * implements java.io.Serializable, Comparable<String>, CharSequence
  67. */
  68. //sorted(Comparator<? super T> comparator); 定制排序 Comparator
  69. Stream<User> stream = users.stream().sorted((e1, e2) -> {
  70. if (e1.getName().equals(e2.getName())) {
  71. return e1.getAge().compareTo(e2.getAge());//倒序取反就行,加个负号 -
  72. } else {
  73. return e1.getName().compareTo(e2.getName());
  74. }
  75. });
  76. stream.forEach(System.out::println);
  77. }
  78. /**
  79. * User{id=1, age=27, name='大青山'}
  80. * User{id=4, age=18, name='池傲天'}
  81. * User{id=2, age=47, name='池寒枫'}
  82. * User{id=3, age=25, name='艾米'}
  83. * User{id=5, age=380, name='霍恩斯'}
  84. * User{id=5, age=500, name='霍恩斯'}
  85. * User{id=5, age=320, name='霍恩斯1'}
  86. * User{id=5, age=600, name='霍恩斯2'}
  87. */
  88. }
  1. public class StreamTest3 {
  2. List<User> users = Arrays.asList(new User(3, 25, "艾米", Arrays.asList("shanghai", "beijing")),
  3. new User(1, 27, "大青山", Arrays.asList("shanghai", "beijing")),
  4. new User(2, 47, "池寒枫", Arrays.asList("shanghai", "beijing")),
  5. // new User(3, 25, "艾米", Arrays.asList("shanghai", "beijing")),
  6. // new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  7. // new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  8. // new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  9. // new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  10. new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  11. // new User(5, 380, "霍恩斯", Arrays.asList("shanghai", "beijing")),
  12. new User(5, 500, "霍恩斯", Arrays.asList("shanghai", "beijing")));
  13. //终止操作
  14. @Test
  15. public void test() {
  16. //anyMatch(Predicate<? super T> predicate) 检查是否至少匹配一个元素
  17. boolean b = users.stream().anyMatch(user -> user.getAge() > 600);
  18. System.out.println(b);//false
  19. //allMatch(Predicate<? super T> predicate) 检查是否匹配所有元素
  20. boolean b1 = users.stream().allMatch(user -> user.getAge() < 600);
  21. System.out.println(b1);//true
  22. //noneMatch(Predicate<? super T> predicate) 检查是否所有元素都没有匹配到
  23. boolean b2 = users.stream().noneMatch(user -> "雷葛".equals(user.getName()));
  24. System.out.println(b2);//true
  25. //先排序再获取第一个最大或最小的值
  26. //Optional<T> findFirst() 返回第一个元素 ,注意因为集合可能为空因此返回了Optional对象
  27. Optional<User> user = users.stream().sorted(Comparator.comparing(User::getAge)).findFirst();
  28. user.ifPresent(System.out::println);
  29. //先过滤出符合条件的随便找一个
  30. // Optional<T> findAny() 返回当前流中的任意元素,总感觉不是返回任意一个?而是返回第一个符合条件的
  31. Optional<User> userOptional = users.stream().filter(u -> u.getAge() < 30).findAny();
  32. userOptional.ifPresent(System.out::println);
  33. long count = users.stream().count();
  34. System.out.println(count);
  35. //Optional<T> max(Comparator<? super T> comparator) 返回流中最大值
  36. Optional<User> max = users.stream().max(Comparator.comparing(User::getAge));
  37. System.out.println(max);
  38. //Optional<T> min(Comparator<? super T> comparator) 返回流中最小值
  39. Optional<String> min = users.stream().map(u -> u.getName()).min(String::compareTo);
  40. System.out.println(min);
  41. }
  42. //归约
  43. @Test
  44. public void test1(){
  45. /**
  46. * 归约,可以将流中的元素反复结合起来,得到一个值。像我们下面可以对各个值进行累加操作,也可以进行累乘等操作
  47. * identity作为起始值作为x,然后把流中的元素作为y进行操作;得到的结果继续作为x,继续对流的元素填充y进行操作
  48. * T reduce(T identity, BinaryOperator<T> accumulator);
  49. *BinaryOperator<T> :R apply(T t, U u);
  50. */
  51. // Integer totalAge = users.stream().map(user -> user.getAge()).reduce(0, (x, y) -> x + y);
  52. //get和累加的两种写法
  53. // Integer totalAge = users.stream().map(User::getAge).reduce(0, Integer::sum);
  54. //如果没有identity作为起始值,那么返回的是Optional,因为集合可能为空
  55. //map-reduce连接称为map-reduce模式,网络搜索模式(大数据搜索)
  56. Optional<Integer> totalAge = users.stream().map(User::getAge).reduce( Integer::sum);
  57. System.out.println(totalAge.get());//617
  58. //归约拼接字符串
  59. Optional<String> stringOptional = users.stream().map(User::getName).reduce(String::concat);
  60. System.out.println(stringOptional.get());//艾米大青山池寒枫池傲天霍恩斯
  61. }
  62. //收集
  63. @Test
  64. public void test2(){
  65. /**
  66. * collect将流转化为其他的形式,接收一个collector实现,用于给stream中元素做汇总的方法
  67. */
  68. List<String> list = users.stream().map(User::getName).collect(Collectors.toList());
  69. list.forEach(System.out::print);//艾米大青山池寒枫池傲天霍恩斯
  70. //set去重
  71. Set<String> set = users.stream().map(User::getName).collect(Collectors.toSet());
  72. set.forEach(System.out::print);//艾米大青山池寒枫池傲天霍恩斯
  73. //把流中元素收集到创建的集合.如HashSet::new
  74. HashSet<String> hashSet = users.stream().map(User::getName).collect(Collectors.toCollection(HashSet::new));
  75. hashSet.forEach(System.out::print);//艾米大青山池寒枫池傲天霍恩斯
  76. //收集统计
  77. Long total = users.stream().collect(Collectors.counting());
  78. System.out.println(total);//5
  79. //平均值。年龄平均值
  80. Double avarage = users.stream().collect(Collectors.averagingInt(User::getAge));
  81. System.out.println(avarage);//123.4
  82. //总和
  83. Integer sum = users.stream().collect(Collectors.summingInt(User::getAge));
  84. System.out.println(sum);//617
  85. //最大值
  86. // Optional<User> max = users.stream().max((u1, u2) -> Integer.compare(u1.getAge(), u2.getAge()));
  87. Optional<User> max = users.stream().collect(Collectors.maxBy((u1, u2) -> Integer.compare(u1.getAge(), u2.getAge())));
  88. System.out.println(max.get());//User{id=5, age=500, name='霍恩斯'}
  89. //最小值.注意不是对象就可以用Integer::compare方法引用替换,对象的话还是要写函数式方法
  90. Optional<Integer> min = users.stream().map(User::getAge).collect(Collectors.minBy(Integer::compare));
  91. System.out.println(min.get());//18
  92. }
  93. //分组
  94. @Test
  95. public void test3(){
  96. //按名字分组
  97. Map<String, List<User>> map = users.stream().collect(Collectors.groupingBy(User::getName));
  98. System.out.println(map);
  99. //{霍恩斯=[User{id=5, age=500, name='霍恩斯'}], 艾米=[User{id=3, age=25, name='艾米'}], 池傲天=[User{id=4, age=18, name='池傲天'}], 大青山=[User{id=1, age=27, name='大青山'}], 池寒枫=[User{id=2, age=47, name='池寒枫'}]}
  100. //多级分组
  101. Map<String, Map<String, List<User>>> collect = users.stream().collect(Collectors.groupingBy(user -> {
  102. if (user.getAge() < 20) {
  103. return "young";
  104. } else if (user.getAge() < 50) {
  105. return "adult";
  106. } else {
  107. return "old";
  108. }
  109. }, Collectors.groupingBy(User::getName)));
  110. System.out.println(collect);
  111. //{young={池傲天=[User{id=4, age=18, name='池傲天'}]}, old={霍恩斯=[User{id=5, age=500, name='霍恩斯'}]}, adult={艾米=[User{id=3, age=25, name='艾米'}], 大青山=[User{id=1, age=27, name='大青山'}], 池寒枫=[User{id=2, age=47, name='池寒枫'}]}}
  112. }
  113. //分区,即满足与不满足条件的元素区分
  114. @Test
  115. public void test4(){
  116. //年龄大于50的为true,满足条件
  117. Map<Boolean, List<User>> map = users.stream().collect(Collectors.partitioningBy(user -> user.getAge() > 50));
  118. System.out.println(map);
  119. //{false=[User{id=3, age=25, name='艾米'}, User{id=1, age=27, name='大青山'}, User{id=2, age=47, name='池寒枫'}, User{id=4, age=18, name='池傲天'}], true=[User{id=5, age=500, name='霍恩斯'}]}
  120. }
  121. @Test
  122. public void test5(){
  123. //获取统计,收集流中Integer属性的统计值。
  124. IntSummaryStatistics statistics = users.stream().collect(Collectors.summarizingInt(User::getAge));
  125. System.out.println(statistics.getAverage());//123.4
  126. System.out.println(statistics.getMax());//500
  127. System.out.println(statistics.getSum());//617
  128. }
  129. //连接流中每个字符串
  130. @Test
  131. public void test6(){
  132. String str = users.stream().map(User::getName).collect(Collectors.joining());
  133. //加分隔符,
  134. String str1 = users.stream().map(User::getName).collect(Collectors.joining("-"));
  135. //加前缀、后缀
  136. String str2 = users.stream().map(User::getName).collect(Collectors.joining("-","start","end"));
  137. System.out.println(str);//艾米大青山池寒枫池傲天霍恩斯
  138. System.out.println(str1);//艾米-大青山-池寒枫-池傲天-霍恩斯
  139. System.out.println(str2);//start艾米-大青山-池寒枫-池傲天-霍恩斯end
  140. }
  141. }
  1. public class StreamPractice {
  2. List<User> users = Arrays.asList(new User(3, 25, "艾米", Arrays.asList("shanghai", "beijing")),
  3. new User(1, 27, "大青山", Arrays.asList("shanghai", "beijing")),
  4. new User(2, 47, "池寒枫", Arrays.asList("shanghai", "beijing")),
  5. new User(4, 18, "池傲天", Arrays.asList("shanghai", "beijing")),
  6. new User(5, 500, "霍恩斯", Arrays.asList("shanghai", "beijing")));
  7. @Test
  8. public void test() {
  9. List<Integer> list = Arrays.asList(1, 3, 5, 8);
  10. List<Integer> collect = list.stream().map(e -> e * e).collect(Collectors.toList());
  11. System.out.println(collect);//[1, 9, 25, 64]
  12. }
  13. @Test
  14. public void test1() {
  15. Optional<Integer> total = users.stream().map(user -> 1).reduce(Integer::sum);
  16. System.out.println(total);//5
  17. }
  18. }

并行流

Fork/Join 框架与传统线程池的区别

采用“工作窃取”模式(work-stealing)

当执行新的任务时它可以将其拆分分成更小的任务执行,并将小任务加到线程队列中,然后再从一个随机线程的队列中偷一个并把它放在自己的队列中。

相对于一般的线程池实现,fork/join框架的优势体现在对其中包含的任务的处理方式上.在一般的线程池中,如果一个线程正在执行的任务由于某些原因无法继续运行,那么该线程会处于等待状态.而在fork/join框架实现中,如果某个子问题由于等待另外一个子问题的完成而无法继续运行.那么处理该子问题的线程会主动寻找其他尚未运行的子问题来执行.这种方式减少了线程的等待时间,提高了性能.

示例:

  1. @Test
  2. public void test2() {
  3. Instant start = Instant.now();
  4. OptionalLong aLong = LongStream.rangeClosed(0, 100000000).parallel().reduce(Long::sum);
  5. System.out.println(aLong);
  6. Instant end = Instant.now();
  7. System.out.println("耗时长:" + Duration.between(start, end).toMillis());
  8. }
  9. //OptionalLong[5000000050000000]
  10. //耗时长:55

Optional——容器类

Optional 类(java.util.Optional) 是一个容器类,代表一个值存在或不存在,原来用null 表示一个值不存在,现在Optional 可以更好的表达这个概念。并且可以避免空指针异常。

示例:

  1. public class OptionalTest {
  2. /**
  3. * 常用方法:
  4. * Optional.of(T t) : 创建一个Optional 实例
  5. * Optional.empty() : 创建一个空的Optional 实例
  6. * Optional.ofNullable(T t):若t 不为null,创建Optional 实例,否则创建空实例
  7. * isPresent() : 判断是否包含值
  8. * orElse(T t) : 如果调用对象包含值,返回该值,否则返回t
  9. * orElseGet(Supplier s) :如果调用对象包含值,返回该值,否则返回s 获取的值
  10. * map(Function f): 如果有值对其处理,并返回处理后的Optional,否则返回Optional.empty()
  11. * flatMap(Function mapper):与map 类似,要求返回值必须是Optional
  12. */
  13. @Test
  14. public void test(){
  15. //Optional.of(T t) : 创建一个Optional 实例
  16. Optional<User> userOptional = Optional.of(new User());
  17. User user = userOptional.get();
  18. System.out.println(user);//User{id=null, age=null, name='null'}
  19. Optional<User> empty = Optional.empty();
  20. // User user1 = empty.get();//java.util.NoSuchElementException: No value present
  21. // System.out.println(user1);
  22. //Optional.ofNullable(T t):若t 不为null,创建Optional 实例,否则创建空实例
  23. //是of和empty方法的组合使用
  24. Optional<User> userOptional2 = Optional.ofNullable(new User());
  25. //isPresent() : 判断是否包含值
  26. if (userOptional2.isPresent()){
  27. System.out.println(userOptional2.get());
  28. }
  29. //ifPresent(Consumer<? super T> consumer) 判断存在包含值,存在则进行Consumer消费,一般选择这个处理Optional
  30. userOptional2.ifPresent(System.out::print);
  31. //orElse(T t) : 如果调用对象包含值,返回该值,否则返回参数值t
  32. User user1 = empty.orElse(new User());
  33. //orElseGet(Supplier s) :如果调用对象包含值,返回该值,否则调用供给型函数获取值,
  34. User user2 = empty.orElseGet(User::new);
  35. //ap(Function f): 如果有值对其处理,并返回处理后的Optional,否则返回Optional.empty()
  36. // Optional<String> s = userOptional2.map(e -> e.getName());
  37. //flatMap要求返回值必须是Optional,因此必须对返回值进行Optional.of包装
  38. Optional<String> optionalS = userOptional2.flatMap(e -> {
  39. return Optional.of(e.getName());
  40. });
  41. }
  42. @Test
  43. public void test1(){
  44. Optional<User> optional = Optional.ofNullable(null);
  45. String fanName = getFanName(optional);
  46. System.out.println(fanName);//null
  47. }
  48. //注意方法参数列表同样可以使用Optional包装
  49. public String getFanName(ser){
  50. //Optional使用Optional封装属性值对象及orElse来避免过多的空指针判断
  51. return user.orElse(new User())
  52. .getFan()
  53. .orElse(new Fan())
  54. .getName();//null
  55. }
  56. }

接口中的默认方法与静态方法

Java 8中允许接口中包含具有具体实现的方法,该方法称为“默认方法”,默认方法使用default关键字修饰。

接口默认方法的”类优先”原则

若一个接口中定义了一个默认方法,而另外一个父类或接口中又定义了一个同名的方法时:

  • 选择父类中的方法。如果一个父类提供了具体的实现,那么接口中具有相同名称和参数的默认方法会被忽略
  • 接口冲突。如果一个父接口提供一个默认方法,而另一个接口也提供了一个具有相同名称和参数列表的方法(不管方法是否是默认方法),那么必须覆盖该方法来解决冲突。覆盖选择其中一个接口的默认实现或者自己重写一个实现。

示例:

  1. public interface IHello {
  2. default void say(){
  3. System.out.println("say hello");
  4. }
  5. String getName();
  6. //Java8 中,接口中允许添加静态方法
  7. public static void staticMethod(String str){
  8. System.out.println(str);
  9. }
  10. }

新时间日期API

LocalDate、LocalTime、LocalDateTime 类的实例是不可变的对象,因此是线程安全的,分别表示使用ISO-8601日历系统的日期、时间、日期和时间。而SimpleDateFormat是可变对象,因此不是线程安全的,在使用时要注意每次使用时new创建一个新的SimpleDateFormat对象或者把SimpleDateFormat放在ThreadLocal上。

注:ISO-8601日历系统是国际标准化组织制定的现代公民的日期和时间的表示法。

Instant 时间戳,用于“时间戳”的运算。它是以Unix元年(传统的设定为UTC时区1970年1月1日午夜时分)开始所经历的描述进行运算

Duration:用于计算两个“时间”间隔

Period:用于计算两个“日期”间隔

Java8 中加入了对时区的支持,带时区的时间为分别为:

ZonedDate、ZonedTime、ZonedDateTime参考地址

代码示例:

  1. public class LocalDateTest {
  2. //LocalDateTime :本地日期时间 LocalDate LocalTime
  3. @Test
  4. public void test(){
  5. //获取当前日期时间
  6. LocalDateTime now = LocalDateTime.now();
  7. System.out.println(now);//2018-12-15T01:40:03.375
  8. //获取指定日期时间
  9. LocalDateTime time = LocalDateTime.of(2018, 7, 10, 10, 10, 59);
  10. System.out.println(time);//2018-07-10T10:10:59
  11. //获取增加一天的日期时间
  12. LocalDateTime nextDay = now.plusDays(1);
  13. System.out.println(nextDay);//2018-12-16T01:41:41.071
  14. //获取减少2个月的日期时间
  15. LocalDateTime time1 = now.minusMonths(2);
  16. System.out.println(time1);//2018-10-15T01:42:43.969
  17. System.out.println(now.getMonth());//DECEMBER
  18. System.out.println(now.getDayOfMonth());//15
  19. System.out.println(now.getMonthValue());//12
  20. System.out.println(now.getSecond());//45
  21. }
  22. //Instant 时间戳
  23. @Test
  24. public void test2(){
  25. //获取当前时间戳,默认获取的是UTC时区的时间戳
  26. Instant now = Instant.now();
  27. System.out.println(now);//2020-12-14T17:58:20.991Z
  28. OffsetDateTime dateTime = now.atOffset(ZoneOffset.ofHours(8));
  29. System.out.println(dateTime);//2020-12-15T02:05:57.558+08:00
  30. OffsetDateTime dateTime1 = now.atOffset(ZoneOffset.of("+8"));
  31. System.out.println(dateTime1);//2020-12-15T02:05:57.558+08:00
  32. //获取当前时间的时间戳
  33. long epochMilli = now.toEpochMilli();
  34. //旧的获取方式
  35. long l = System.currentTimeMillis();
  36. //新老获取时间戳的值是一样的,只是新的时间戳返回的时间是UTC时区的日期时间,但在用new Date(l)转化时或默认转化为
  37. //当前日期时间。这个需要注意
  38. System.out.println(epochMilli);//1607968605927
  39. System.out.println(l);//1607968605927
  40. System.out.println(new Date(l));//Tue Dec 15 01:59:01 CST 2020
  41. }
  42. //Duration:时间间隔
  43. @Test
  44. public void test3(){
  45. Instant now = Instant.now();
  46. Instant instant = Instant.ofEpochMilli(1607968605927L);
  47. //Duration:用于计算两个“时间”间隔.注意间隔是用第二个参数减去第一个参数的间隔,因此当第一个参数是比较晚时得到的是负数
  48. Duration duration = Duration.between(instant, now);
  49. //获取间隔的毫秒值用to
  50. System.out.println(duration.toMillis());//992143
  51. //获取间隔的秒值用get
  52. System.out.println(duration.getSeconds());//992
  53. LocalDateTime localDateTime = LocalDateTime.now();
  54. LocalDateTime dateTime = LocalDateTime.of(2020, 12, 15, 0, 0, 0);
  55. Duration between = Duration.between(localDateTime, dateTime);
  56. System.out.println(between.getSeconds());//-8261
  57. }
  58. //Period:日期间隔
  59. @Test
  60. public void test5(){
  61. LocalDate now = LocalDate.now();
  62. LocalDate date = LocalDate.of(2020, 11, 11);
  63. //Period:用于计算两个“日期”间隔 获取值不好用
  64. Period period = Period.between(date, now);
  65. System.out.println(period);//P1M4D:表示间隔1个月04天
  66. System.out.println(period.getChronology());//ISO 年表
  67. System.out.println(period.getYears());//0
  68. System.out.println(period.getMonths());//1
  69. System.out.println(period.getDays());//4
  70. }
  71. //TemporalAdjuster:时间校正器
  72. @Test
  73. public void test6(){
  74. LocalDateTime now = LocalDateTime.now();
  75. //调整日期为该月的第一天
  76. LocalDateTime dateTime = now.withDayOfMonth(1);
  77. System.out.println(dateTime);//2020-12-01T02:39:06.111
  78. //调整日期为该月的第一天
  79. LocalDateTime time = now.with(TemporalAdjusters.firstDayOfMonth());
  80. System.out.println(time);//2020-12-01T02:35:01.715
  81. //调整日期为下一个星期天
  82. LocalDateTime nextSunday = now.with(TemporalAdjusters.next(DayOfWeek.SUNDAY));
  83. System.out.println(nextSunday);//2020-12-20T02:37:01.322
  84. //自定义时间校正器 :获取下一个工作日
  85. LocalDateTime nextWorkDate = now.with(temporal -> {
  86. LocalDateTime dateTime1 = (LocalDateTime) temporal;
  87. if (dateTime1.getDayOfWeek().equals(DayOfWeek.FRIDAY)) {
  88. return dateTime1.plusDays(3);
  89. } else if (dateTime1.getDayOfWeek().equals(DayOfWeek.SATURDAY)) {
  90. return dateTime1.plusDays(2);
  91. } else {
  92. return dateTime1.plusDays(1);
  93. }
  94. });
  95. System.out.println(nextWorkDate);//2020-12-16T02:44:45.370
  96. }
  97. //DateTimeFormatter :格式化时间/日期
  98. @Test
  99. public void test7(){
  100. DateTimeFormatter formatter = DateTimeFormatter.ISO_DATE;
  101. DateTimeFormatter formatter2 = DateTimeFormatter.ofPattern("yyyy-MM-dd 时HH分mm秒ss");
  102. LocalDateTime now = LocalDateTime.now();
  103. //日期转指定格式的字符串
  104. String str = formatter.format(now);
  105. System.out.println(str);//2020-12-15
  106. String str2 = formatter2.format(now);
  107. System.out.println(str2);//2020-12-15 时02分54秒47
  108. //注意2020-12-15这种类型的字符串没法转化为日期时间LocalDateTime,只能转化为日期LocalDate
  109. // 会报错:java.time.format.DateTimeParseException: Text '2020-12-15' could not be parsed: Unable to obtain LocalDateTime from TemporalAccessor: {},ISO resolved to 2020-12-15 of type java.time.format.Parsed
  110. // LocalDateTime dateTime = now.parse(str, formatter);
  111. LocalDate date = LocalDate.parse(str, formatter);
  112. System.out.println(date);//2020-12-15
  113. //指定格式字符串转日期时间
  114. LocalDateTime dateTime2 = LocalDateTime.parse(str2, formatter2);
  115. System.out.println(dateTime2);//2020-12-15T03:01:43
  116. }
  117. //带时区的时间为分别为:ZonedDate、ZonedTime、ZonedDateTime
  118. //Asia/Shanghai America/New_York
  119. @Test
  120. public void test9(){
  121. //获取带时区的日期和时间 方式一
  122. //默认时区
  123. ZonedDateTime now1 = ZonedDateTime.now();
  124. System.out.println(now1);//2020-12-15T03:34:44.479+08:00[Asia/Shanghai]
  125. //用指定时区获取当前时间
  126. ZonedDateTime newyorkNow = ZonedDateTime.now(ZoneId.of("America/New_York"));
  127. System.out.println(newyorkNow);//2020-12-14T14:36:21.999-05:00[America/New_York]
  128. //获取带时区的日期和时间 方式二
  129. //以这种方式创建的ZonedDateTime,它的日期和时间与LocalDateTime相同,但附加的时区不同,因此是两个不同的时刻
  130. LocalDateTime now = LocalDateTime.now(ZoneId.of("America/New_York"));
  131. System.out.println(now.atZone(ZoneId.systemDefault()));//2020-12-14T14:39:04.255+08:00[Asia/Shanghai]
  132. LocalDateTime now2 = LocalDateTime.now(ZoneId.of("America/New_York"));
  133. //获取到的区域时间是当地日期时间 +时区时间差
  134. ZonedDateTime zonedDateTime = now2.atZone(ZoneId.of("America/New_York"));
  135. System.out.println(zonedDateTime);//2020-12-14T14:19:41.923-05:00[America/New_York]
  136. //时区转换
  137. //通过withZoneSameInstant()将关联时区转换到另一个时区,转换后日期和时间都会相应调整。
  138. //时区转换的时候,由于夏令时的存在,不同的日期转换的结果很可能是不同的。
  139. //涉及到时区时,千万不要自己计算时差,否则难以正确处理夏令时。
  140. //默认时区
  141. ZonedDateTime now3 = ZonedDateTime.now();
  142. ZonedDateTime newYorkDateTime1 = now3.withZoneSameInstant(ZoneId.of("America/New_York"));
  143. System.out.println(now3);//2020-12-15T03:42:58.940+08:00[Asia/Shanghai]
  144. System.out.println(newYorkDateTime1);//2020-12-14T14:42:58.940-05:00[America/New_York]
  145. //纽约时区时间转换为本地时间
  146. LocalDateTime localDateTime = newYorkDateTime1.toLocalDateTime();
  147. System.out.println(localDateTime);//丢弃了时区信息
  148. //练习:某航线从北京飞到纽约需要13小时20分钟,请根据北京起飞日期和时间计算到达纽约的当地日期和时间。
  149. ZonedDateTime now4 = ZonedDateTime.now();
  150. ZonedDateTime zonedDateTime1 = now4.plusHours(13).plusMinutes(20);
  151. ZonedDateTime newyorkDateTime2 = zonedDateTime1.withZoneSameInstant(ZoneId.of("America/New_York"));
  152. System.out.println(newyorkDateTime2);//2020-12-15T04:12:39.611-05:00[America/New_York]
  153. System.out.println("before:"+now4);//before:2020-12-15T03:52:39.611+08:00[Asia/Shanghai]
  154. System.out.println("after:"+zonedDateTime1);//after:2020-12-15T17:12:39.611+08:00[Asia/Shanghai]
  155. }
  156. @Test
  157. public void test8(){
  158. Set<String> zoneIds = ZoneId.getAvailableZoneIds();
  159. zoneIds.forEach(System.out::println);
  160. }
  161. }

重复注解和类型注解

示例:

  1. //定义可重复注解的容器
  2. @Target({ElementType.ANNOTATION_TYPE, ElementType.METHOD,ElementType.TYPE})
  3. @Retention(RetentionPolicy.RUNTIME)
  4. public @interface NameAnnotions {
  5. NameAnnotion[] value();
  6. }
  7. //可重复注解必须加@Repeatable(NameAnnotions.class),NameAnnotions为其容器
  8. @Repeatable(NameAnnotions.class)
  9. //注意类型注解要求这两个目标注解都要标注才能使用,否则报错:ElementType.PARAMETER,ElementType.TYPE_PARAMETER
  10. @Target({ElementType.ANNOTATION_TYPE, ElementType.METHOD,ElementType.TYPE,ElementType.PARAMETER,ElementType.TYPE_PARAMETER})
  11. @Retention(RetentionPolicy.RUNTIME)
  12. public @interface NameAnnotion {
  13. String value() default "";
  14. }
  15. public class AnnotionTest {
  16. @Test
  17. public void test() throws NoSuchMethodException {
  18. Class<AnnotionTest> aClass = AnnotionTest.class;
  19. Method work = aClass.getMethod("work");
  20. NameAnnotion[] annotations = work.getAnnotationsByType(NameAnnotion.class);
  21. for (NameAnnotion annotation : annotations) {
  22. System.out.println(annotation.value());
  23. }
  24. }
  25. //可重复注解
  26. @NameAnnotion("996")
  27. @NameAnnotion("955")
  28. public void work(@NameAnnotion String str) {
  29. }
  30. }

其他新特性

HashMap的新特性——从数组加链表的存储方式转变为数组加(链表/红黑树)的方式。

HashMap在存储数据的时候会先计算出Key的hash值,再通过运算得到我们底层存储的数组的下标索引值。 当hash值一样时,会调用equal()方法比较内容是否相等,如果相等则替换原有的值,如果不相等,此时就产生了hash碰撞,数据会被以链表的形式存放起来。而当hash碰撞严重时,比如极端情况下,链表就会很长,导致查询或插入比较时效率低下,这时候我们就引入了采用红黑树来替换链表的形式进行存储。

链表转红黑树的条件是:当链表的长度大于8且HashMap的容量大于64时会触发由链表转红黑树逻辑。红黑树除了添加以外其他的效率都很高,因为链表添加的时候是添加在末尾比较快,而红黑树添加时要进行比较大小添加。

加载因子设置为0.75的原因,因为如果太小,HashMap就会不断扩容,浪费效率。太大了可能插入的值就是一直产生碰撞形成了链表,没有插入到数组的索引位置,导致一直没有扩容,效率低下。

ConcurrentHashMap的新特性——把锁分段技术改成CAS算法

ConcurrentHashMap在JDK1.8之前是采用锁分段技术,默认由16个分段,对于16把锁。因为分段的数量不好控制,如果分段过多就会浪费空间,因为很多段里面其实没有数据进入,而太小也不好,这样就导致效率太低,因为会导致太多操作竞争同一个段锁。

底层也采用数组加(链表/红黑树)的方式提高效率。

方法区实现从永久代实现变为元空间MetaSpace。

元空间使用的是物理内存。默认物理内存有多大,元空间就可以是多大,只受限于物理内存的大小,当然我们也可以指定MetaSpaceSize指定元空间大小,MaxMetaSpaceSize指定最大元空间大小。取代了永久代的PermGenSize和MaxPermGenSize.

Java 8 新特性——实践篇的更多相关文章

  1. Java 8新特性-4 方法引用

    对于引用来说我们一般都是用在对象,而对象引用的特点是:不同的引用对象可以操作同一块内容! Java 8的方法引用定义了四种格式: 引用静态方法     ClassName :: staticMetho ...

  2. Spring 4支持的Java 8新特性一览

    有众多新特性和函数库的Java 8发布之后,Spring 4.x已经支持其中的大部分.有些Java 8的新特性对Spring无影响,可以直接使用,但另有些新特性需要Spring的支持.本文将带您浏览S ...

  3. java 8 新特性

    最近在IDEA的️驱使下,看了点java8的东西,链接贴一下,,,,, 1.Java 8新特性概述2.Java 8中的 Stream API 详解[3.Java 8新特性终极指南] 简单的使用看完新特 ...

  4. Java 8 新特性终极版

    声明:本文翻译自Java 8 Features Tutorial – The ULTIMATE Guide,翻译过程中发现并发编程网已经有同学翻译过了:Java 8 特性 – 终极手册,我还是坚持自己 ...

  5. Java 8新特性前瞻

    快端午小长假了,要上线的项目差不多完结了,终于有时间可以坐下来写篇博客了. 这是篇对我看到的java 8新特性的一些总结,也是自己学习过程的总结. 几乎可以说java 8是目前为止,自2004年jav ...

  6. Java 8新特性探究(八)精简的JRE详解

    http://www.importnew.com/14926.html     首页 所有文章 资讯 Web 架构 基础技术 书籍 教程 Java小组 工具资源 - 导航条 - 首页 所有文章 资讯 ...

  7. Java 8 新特性1-函数式接口

    Java 8 新特性1-函数式接口 (原) Lambda表达式基本结构: (param1,param2,param3) -> {代码块} 例1: package com.demo.jdk8; i ...

  8. 【整理】Java 8新特性总结

    闲语: 相比于今年三月份才发布的Java 10 ,发布已久的Java 8 已经算是老版本了(传闻Java 11将于9月25日发布....).然而很多报道表明:Java 9 和JJava10不是 LTS ...

  9. 一小时上手Java 8新特性

    一小时上手Java 8新特性 本文摘译自 https://www.journaldev.com/2389/java-8-features-with-examples,并做了适当增补. Iterable ...

随机推荐

  1. [USACO14JAN]Ski Course Rating G

    题目链接:https://www.luogu.com.cn/problem/P3101 Slove 这题我们可以尝试建立一个图. 以相邻的两个点建边,边的权值为两个点高度差的绝对值,然后把边按照边权值 ...

  2. composer的autoload来自动加载自己编写的函数库与类库?

    1.使用命令composer init生成composer.json文件,并编辑autoload选项内容如下: 其中又包含主要的两个选项: files 和 psr-4. files就是需要compos ...

  3. ifconfig结果说明

  4. 类虚拟机软件CrossOver是什么?它的优势在哪里?

    虚拟机软件对于很多人来说已经不是一个陌生的词汇了.我们可以通过软件来模拟具有完整硬件系统功能的计算机系统.比如我们可以在Mac OS系统上模拟Windows 7 的系统,以此来安装我们想要使用的应用程 ...

  5. Ubuntu sudo 出现unable to resolve host 解决方法

    Ubuntu sudo 出现unable to resolve host 解决方法 Ubuntu环境, 假设这台机器名字叫abc(机器的hostname), 每次执行sudo 就出现这个警告讯息: s ...

  6. zabbix地图显示全国延迟

    Zabbix 地图显示全国延迟 1.  效果图 2.  实现方法 将地图.png上传到zabbix为背景,上传红绿点.png为图标.然后新建主机关联模板为ICMP Ping,新建一个拓扑图调用地图为背 ...

  7. svn学习与应用

    先来认识下svn svn是之前公司一直在用的代码版本控制系统,采用了分支管理系统.顾名思义,可以对代码的版本做系统化管理.通俗讲就是可用于多个人共同开发同一个项目,实现共用资源的目的. 开发同学使用s ...

  8. C构造类型 数组

    一.数组是什么 一组有个固定大小.相同数据类型的数据的集合. 数组可以分为:一维数组.二维数组.三维数组....(维数不止一维:多维数组) 表格 : 行和列(二维数组) 二.一维数组 1.一般形式 类 ...

  9. dubbo协议之编码请求对象体

    上节我们看了如何编码请求头,这节一起看下过程中,对请求对象的编码,涉及对接口,方法,方法参数类型,方法参数进行编码,DubboCodec中重写了这个方法: request.getData向下转型成Rp ...

  10. 「刷题笔记」哈希,kmp,trie

    Bovine Genomics 暴力 str hash+dp 设\(dp[i][j]\)为前\(i\)组匹配到第\(j\)位的方案数,则转移方程 \[dp[i][j+l]+=dp[i-1][j] \] ...