java8新特性
1. 接口的默认方法
1.1 传统的方法
在Java8之前Java中接口里面的方法默认都是public abstract 修饰的抽象方法并且没有方法体
1.2 static方法
- 使用static修饰接口中的方法并且必须有主体;
- 接口的static方法只能够被接口本身调用;接口名.方法名(…)
- 接口的static方法不能够被子接口继承;
- 接口的static方法不能够被实现类覆写及直接调用
1.3 default方法
在接口中可以定义一个使用default修饰有方法体的方法,接口中可以对这个方法提供默认的一种实现。
- 使用default修饰接口中的方法并且必须有主体;
- 接口的default方法不能够被接口本身调用,需要接口的实例(实现类对象)来调用;
- 接口的default方法可以被子接口继承、覆写或者[实现类对象]直接调用;
- 接口的default方法可以被实现类覆写及直接调用;
2. 函数式接口
2.1 概念
函数式接口(Functional Interface)就是一个有且仅有一个抽象方法[的接口],但是可以有多个非抽象方法的接口。函数式接口可以被隐式转换为lambda表达式
2.2 函数式接口注解
@FunctionalInterface
我们在函数式接口上面加上此注解后,里面就只能够有一个抽象方法了,当然不加此注解且只有一个抽象方法的接口也是函数式接口,只是没有限定提示而已。
3. Lambda表达式
3.1 概念
可以看成是对匿名内部类的简写,使用Lambda表达式时,接口必须是函数式接口。
3.2 语法
基本语法:
<函数式接口> <变量名> = (参数1,参数2…) -> {
//方法体
}
特点说明:
(参数1,参数2…)表示参数列表;->表示连接符;{}内部是方法体
1、=右边的类型会根据左边的函数式接口类型自动推断;
2、如果形参列表为空,只需保留();
3、如果形参只有1个,()可以省略,只需要参数的名称即可;
4、如果执行语句只有1句,且无返回值,{}可以省略,若有返回值,则若想省去{},则必须同时省略return,且执行语句也保证只有1句;
5、形参列表的数据类型会自动推断;
6、lambda不会生成一个单独的内部类文件;
7、lambda表达式若访问了局部变量,则局部变量必须是final的,若是局部变量没有加final关键字,系统会自动添加,此后在修改该局部变量,会报错;
3.3 示例
//省略参数类型
MathInterface m3 = (a,b)->{
return a+b;
};
//当方法体中只有一句代码的时候 {}可以省略 如果由返回值 return一起省略
MathInterface m4 = (a,b)->a+b;
//如果只有一个参数的时候 可以省略类型和圆括号
MathInterface m5=name->{
System.out.println(name+"666");
};
//没有参数的时候不能省略圆括号
MathInterface m7=()->System.out.println("今天中午吃啥");
4. 方法引用
4.1 构造方法引用
函数式接口的名称 变量名=类名::new;(会去调用类中对应的构造方法)
4.2 静态方法引用
函数式接口的名称 变量名=类名::方法名;
4.3 实例方法引用
使用java自带的Function接口
函数式接口的名称 变量名=对象::方法名;
5. Stream流
5.1 概念
Stream(流)是一个来自数据源的元素队列并支持聚合操作,主要式操作集合,数组,I/O channel, 产生器generator 等
5.2 生成流
在 Java 8 中, 集合接口有两个方法来生成流:
stream() − 为集合创建串行流。
parallelStream() − 为集合创建并行流。
5.3 常用方法
forEach
Stream 提供了新的方法 ‘forEach’ 来迭代流中的每个数据。以下代码片段使用 forEach 输出了10个随机数:
Random random = new Random();
random.ints().limit(10).forEach(System.out::println);
map
map 方法用于映射每个元素到对应的结果,以下代码片段使用 map 输出了元素对应的平方数:
List<Integer> numbers = Arrays.asList(3, 2, 2, 3, 7, 3, 5);
// 获取对应的平方数
List<Integer> squaresList = numbers.stream().map( i -> i*i).distinct().collect(Collectors.toList());
filter
filter 方法用于通过设置的条件过滤出元素。以下代码片段使用 filter 方法过滤出空字符串:
List<String>strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl"); // 获取空字符串的数量
int count = strings.stream().filter(string -> string.isEmpty()).count();
limit
limit 方法用于获取指定数量的流。 以下代码片段使用 limit 方法打印出 10 条数据:
Random random = new Random();
random.ints().limit(10).forEach(System.out::println);
sorted
Random random = new Random();
random.ints().limit(10).sorted().forEach(System.out::println);
并行(parallel)程序
List<String> strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl");
// 获取空字符串的数量
int count = strings.parallelStream().filter(string -> string.isEmpty()).count();
Collectors
Collectors 类实现了很多归约操作,例如将流转换成集合和聚合元素。Collectors 可用于返回列表或字符串:
List<String>strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl");
List<String> filtered = strings.stream().filter(string -> !string.isEmpty()).collect(Collectors.toList());
System.out.println("筛选列表: " + filtered);
String mergedString = strings.stream().filter(string -> !string.isEmpty()).collect(Collectors.joining(", "));
System.out.println("合并字符串: " + mergedString);
6. Optional 类
Optional 类是一个可以为null的容器对象。如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象。
Optional 是个容器:它可以保存类型T的值,或者仅仅保存null。Optional提供很多有用的方法,这样我们就不用显式进行空值检测。
Optional 类的引入很好的解决空指针异常
善用Optional可以使我们代码中很多繁琐、丑陋的设计变得十分优雅。
6.1 示例
我们可以通过以下实例来更好的了解 Optional 类的使用:
以往的判断方式:
引入Optional错误的判断方式
正确的判断方式:
7. 日期时间API
7.1 本地日期时间 API
LocalDate/LocalTime 和 LocalDateTime 类可以在处理时区不是必须的情况。代码如下
// 获取当前的日期时间
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);
7.2 时区的日期时间API
如果我们需要考虑到时区,就可以使用时区的日期时间API
// 获取当前时间日期
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);