阅读 47

Java8新特性

自从使用上了Java8的Lambda和Stream编程后,彻底的爱上了。因此决定研究下其他新特性。

Lambda表达式

什么是lambda表达式?lambda表达式就是一个匿名函数,是一段可以传递的代码。表达式表达式,你可以理解为就是一堆表达式以函数形式封装在一起,只不过没有函数名。这堆表达式可以被一个引用持有,传递这个引用就是传递这个匿名函数。

但是我们知道,Java中所有的东西都是类或者接口,那lambda应该属于类还是接口呢?答案就是lambda属于函数式接口,就是只有一个方法的接口。只有接口只有一个方法,那么他就可以用lambda来表示,比如Runnable、Callable、Comparator接口。Java8还新增了几个通用的函数式接口:Function、Comsumer、Supplier、Predicate。

  • Function:函数型接口,属于一个入参,输出一出参
  • Consumer接口:消费型接口,输入一个入参,无返回值,形象的理解就是把参数吃(消费)了
  • Supplier:供给型接口,无入参,返回一个值。形象理解就是你不用给我提供什么东西,我供给你一些东西。
  • Predicate:断言型接口,接收一个参数,返回一个boolean值。

lambda表达式的作用有:

  • 简化部分匿名内部类的写法:为什么是部分呢?因为lambda只能简化接口只有函数式接口的匿名内部类。函数式接口只有一个未实现的函数,你使用lambda匿名函数就是实现该接口。如果匿名内部类有多个未实现的方法,当然就不能使用lambda表达式类取代了。

综上,lambda就是一个函数式接口,可以自己写一个函数式接口来接收lambda,或者使用Java提供的通用的接口Function、Consumer、Supplier、Predicate接口。

函数式接口

就是只有一个方法的接口,可以使用@FunctionInterface注解加到接口上,约束该接口不能超过两个方法。当然不加该注解也可以。

方法引用与构造器引用

其实就是lambda的更进一步的简写。

方法引用:和lambda类似,写lambda表达式时需要你手动写一堆表达式,如果这堆表达式其实某个方法已经实现了,则不需要再写lambda了,只需要传入方法引用即可。主要有三种表达方式:

  • 对象::实例方法名
  • 类::静态方法名
  • 类::实例方法名:上面都是把参数对应到参数列表,这种方式则不是,本来方法参数需要和参数列表对应,但如果方法的第一个参数是该参数实例,则可以用这种方法。比如,你如果想写 (a) -> a.getName(); 则完全可以写为A::getName

例如你要写一个大于String的lambda,一般是:

Consumer<String> c = x -> System.out.println(x);
复制代码

你可以发现,这个打印的动作是你手动写表达式做的,即System.out.println(x),其实打印的动作System.out这个实例的println已经实现了,那么可以使用方法引用如下:

Consumer<String> c = System.out::println;
复制代码

构造器引用,类名::new。用来生成一个对象。如果一个类有多个构造器,则new选用哪个构造器取决于方法入参。

如果要调用无参构造器,则用供给型接口:

Supplier<Employee> s = Employee::new;
复制代码

如果要调用单个参数的构造器,则使用函数型接口:

Function<Integer, Employee> f = Employee::new;
复制代码

由上可知,一个引用可以对应到不同的重载方法,具体使用的是哪个方法,要看你接收的函数型接口的参数。

Stream API

流式操作,Stream不会改变源集合,通过Stream操作后会生成流。标准的Stream操作是:

  • 通过集合创建Stream流,一般是.stream()方法
  • 做操作
  • 终止操作,生成新集合

Optional类

自我感觉不好用,不做介绍,这是用来避免空指针的,我觉得还不如用Spring的Assert来断言比较好

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

Java8后,接口中可以有已经实现的默认方法。接口中也可以有静态实现的方法。

新时间日期API

Java8以前的日期Date不易使用,同时日期格式化类SimpleDateFormatter存在线程安全问题。Java8新出的几个日期类型则非常的易用

  • LocalDate:日期类型
  • LocalTime:时间类型
  • LocalDateTime:日期时间类型
  • Instant:时间戳,of方法是从1970-01-01开始算起
  • Duration:计算两个时间之间的间隔,Duration.between(LocalTime, LocalTime)
  • Period:计算两个日期之间的间隔,Period.between(LocalDate, LocalDate)
  • TemporalAdjuster:时间校正器,如,获取下一个周六,LocalDateTime.now().with(TemporalAdjusters.next(DayOfWeek.SATURDAY))
  • DateTimeFormatter:时间日期格式化
  • ZoneId:时区转化