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:时区转化