Java8 新特性 学习笔记

Java8 新特性

1 新特性

Java8 新增了非常多的特性,主要有:

  • Lambda 表达式 − Lambda允许把函数作为一个方法的参数(函数作为参数传递进方法中)。
  • 方法引用 − 方法引用提供了非常有用的语法,可以直接引用已有Java类或对象(实例)的方法或构造器。与lambda联合使用,方法引用可以使语言的构造更紧凑简洁,减少冗余代码。
  • 默认方法 − 默认方法就是一个在接口里面有了一个实现的方法。
  • 新工具 − 新的编译工具,如:Nashorn引擎 jjs、 类依赖分析器jdeps。
  • Stream API −新添加的Stream API(java.util.stream) 把真正的函数式编程风格引入到Java中。
  • Date Time API − 加强对日期与时间的处理。
  • Optional 类 − Optional 类已经成为 Java 8 类库的一部分,用来解决空指针异常。
  • Nashorn, JavaScript 引擎 − Java 8提供了一个新的Nashorn javascript引擎,它允许我们在JVM上运行特定的javascript应用。

2 Java8 编程风格初窥

package test.testJava8;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;

/**
 * fileName: Java8Tester
 * description: Java7 与 java8 比较
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 9:09
 */
public class Java8Tester {

    /**
     * 使用Java7来对list排序
     *
     * @param names
     */
    private void sortByJava7(List<String> names) {
        Collections.sort(names, new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                return o1.compareTo(o2);
            }
        });
    }

    /**
     * 使用Java8来对list进行排序
     * 
     * @param names
     */
    private void sortByJava8(List<String> names) {
        Collections.sort(names, (o1, o2) -> o1.compareTo(o2));
    }

    public static void main(String[] args) {
        List<String> names = new ArrayList<>();

        names.add("zhangsan");
        names.add("lisi");
        names.add("wanger");

        List<String> names1 = new ArrayList<>();

        names1.add("zhangsan");
        names1.add("lisi");
        names1.add("wanger");


        Java8Tester java8Tester = new Java8Tester();

        System.out.println("使用java7:");
        java8Tester.sortByJava7(names);
        System.out.println(names);

        System.out.println("使用java8:");
        java8Tester.sortByJava8(names1);
        System.out.println(names1);

    }
}

3 新特性之Lambda表达式

Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中)。

使用 Lambda 表达式可以使代码变的更加简洁紧凑。

3.1 语法
(parameters) -> expression
或
(parameters) -> { statements; }

特征:

  • 可选类型声明: 不需要声明参数类型,编译器可以统一识别参数值。
  • 可选的参数圆括号: 一个参数无需定义圆括号,但多个参数需要定义圆括号。
  • 可选的大括号: 如果主体包含了一个语句,就不需要使用大括号。
  • 可选的返回关键字: 如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定明表达式返回了一个数值。
3.2 Lambda表达式实例
// 1. 不需要参数,返回值为 5  
() -> 5  
  
// 2. 接收一个参数(数字类型),返回其2倍的值  
x -> 2 * x  
  
// 3. 接受2个参数(数字),并返回他们的差值  
(x, y) -> x – y  
  
// 4. 接收2个int型整数,返回他们的和  
(int x, int y) -> x + y  
  
// 5. 接受一个 string 对象,并在控制台打印,不返回任何值(看起来像是返回void)  
(String s) -> System.out.print(s)
package test.testJava8;

/**
 * fileName: LambdaTest
 * description: Lambda测试
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 9:40
 */
public class LambdaTest {

    // 运算接口
    interface MathOperation{
        int operation(int a, int b);
    }

    // service接口
    interface GreetingService{
        void sayMessage(String msg);
    }

    // 成员方法
    private int operate(int a, int b, MathOperation operation) {
        return operation.operation(a, b);
    }
    
    public static void main(String[] args) {
        LambdaTest test = new LambdaTest();

        // 定义方法
        MathOperation addition=(a, b) -> a+b;
        System.out.println("a+b=" + test.operate(1, 2, addition));

        // 定义方法
        GreetingService service = msg -> System.out.println("hello " + msg);
        // 调用方法
        service.sayMessage("lihaogn");
    }
}

3.3 变量作用域
  • lambda 表达式只能引用标记了 final 的外层局部变量,这就是说不能在 lambda 内部修改定义在域外的局部变量,否则会编译错误。
  • lambda 表达式的局部变量可以不用声明为 final,但是必须不可被后面的代码修改(即隐性的具有 final 的语义)。
  • 在 Lambda 表达式当中不允许声明一个与局部变量同名的参数或者局部变量。

4 新特性之方法引用

方法引用通过方法的名字来指向一个方法。

方法引用可以使语言的构造更紧凑简洁,减少冗余代码。

方法引用使用一对冒号 ::

package test.testJava8;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Supplier;

/**
 * fileName: MethodReferenceTest
 * description: 方法引用实例
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 9:59
 */
public class Car {

    public static Car create(final Supplier<Car> supplier) {
        return supplier.get();
    }

    /**
     * collide-->相撞
     *
     * @param car
     */
    public static void collide(final Car car) {
        System.out.println("collided: " + car.toString());
    }

    public void follow(final Car another) {
        System.out.println("following the " + another.toString());
    }

    public void repair() {
        System.out.println("repaired " + this.toString());
    }

    public static void main(String[] args) {

        // 构造器引用
        final Car car = Car.create(Car::new);
        final List<Car> cars = Arrays.asList(car);

        // 静态方法引用
        cars.forEach(Car::collide);
        // 类的任意对象方法引用
        cars.forEach(Car::repair);
        // 特定对象方法的应用
        final Car police = Car.create(Car::new);
        cars.forEach(police::follow);

        List<String> names = new ArrayList();

        names.add("Google");
        names.add("Runoob");
        names.add("Taobao");
        names.add("Baidu");
        names.add("Sina");

        names.forEach(System.out::println);
    }
}

5 新特性之函数式接口

函数式接口(Functional Interface)就是一个有且仅有一个抽象方法,但是可以有多个非抽象方法的接口。

函数式接口可以被隐式转换为 lambda 表达式。

5.1 实例

Predicate <T> 接口是一个函数式接口,它接受一个输入参数 T,返回一个布尔值结果。

该接口包含多种默认方法来将Predicate组合成其他复杂的逻辑(比如:与,或,非)。

package test.testJava8;

import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;

/**
 * fileName: FunctionInterfaceTest
 * description: 函数式接口实例
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 10:14
 */
public class FunctionInterfaceTest {

    public static void eval(List<Integer> list, Predicate<Integer> predicate) {
        for (Integer n : list) {
            if (predicate.test(n)) {
                System.out.print(n + " ");
            }
        }

        // java8写法
//        list.stream().filter(predicate).forEach(System.out::print);
    }

    public static void main(String[] args) {
        List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8);

        System.out.println("输出所有数据");
        // n存在则返回true
        eval(list, n -> true);
        System.out.println();
        System.out.println("输出所有偶数");
        // n是偶数返回true
        eval(list, n -> n % 2 == 0);
        System.out.println();
        System.out.println("输出所有大于3的数");
        // n大于3返回true
        eval(list, n -> n > 3);

    }
}
5.2 补充
  • 关于 @FunctionalInterface 注解

    Java 8为函数式接口引入了一个新注解@FunctionalInterface,主要用于编译级错误检查,加上该注解,当你写的接口不符合函数式接口定义的时候,编译器会报错。

  • 函数式接口里允许定义默认方法

  • 函数式接口里允许定义静态方法

  • 函数式接口里允许定义 java.lang.Object 里的 public 方法

6 新特性之默认方法

默认方法就是接口可以有实现方法,而且不需要实现类去实现其方法。

我们只需在方法名前面加个 default 关键字即可实现默认方法。

为什么要有这个特性?

首先,之前的接口是个双刃剑,好处是面向抽象而不是面向具体编程,缺陷是,当需要修改接口时候,需要修改全部实现该接口的类。

6.1 语法
public interface Vehicle {
   default void print(){
      System.out.println("我是一辆车!");
   }
}
多个默认方法
public interface Vehicle {
   default void print(){
      System.out.println("我是一辆车!");
   }
}
 
public interface FourWheeler {
   default void print(){
      System.out.println("我是一辆四轮车!");
   }
}

一个接口有默认方法,考虑这样的情况,一个类实现了多个接口,且这些接口有相同的默认方法,有两种解决方法:

  • 创建自己的默认方法,来覆盖重写接口的默认方法:

    public class Car implements Vehicle, FourWheeler {
       default void print(){
          System.out.println("我是一辆四轮汽车!");
       }
    }
    
  • 使用 super 来调用指定接口的默认方法:

    public class Car implements Vehicle, FourWheeler {
       public void print(){
          Vehicle.super.print();
       }
    }
    
6.2 静态默认方法
public interface Vehicle {
   default void print(){
      System.out.println("我是一辆车!");
   }
    // 静态方法
   static void blowHorn(){
      System.out.println("按喇叭!!!");
   }
}

7 新特性之stream

stream将要处理的元素集合看作一种流, 流在管道中传输, 并且可以在管道的节点上进行处理, 比如筛选, 排序,聚合等。

7.1 steam

Stream(流)是一个来自数据源的元素队列并支持聚合操作。

  • 元素是特定类型的对象,形成一个队列。 Java中的Stream并不会存储元素,而是按需计算。
  • 数据源: 流的来源。 可以是集合,数I/O channel, 产生器generator 等。
  • 聚合操作:类似SQL语句一样的操作, 比如filter, map, reduce, find, match, sorted等。

和以前的Collection操作不同, Stream操作还有两个基础的特征:

  • Pipelining:中间操作都会返回流对象本身。 这样多个操作可以串联成一个管道, 如同流式风格(fluent style)。 这样做可以对操作进行优化, 比如延迟执行(laziness)和短路( short-circuiting)。
  • 内部迭代:以前对集合遍历都是通过Iterator或者For-Each的方式, 显式的在集合外部进行迭代, 这叫做外部迭代。 Stream提供了内部迭代的方式, 通过访问者模式(Visitor)实现。
7.2 使用
1 生成流

两种方法:

  • stream() − 为集合创建串行流。
  • parallelStream() − 为集合创建并行流。
List<String> strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl");
List<String> filtered = strings.stream()
    .filter(string -> !string.isEmpty()).collect(Collectors.toList());
2 foreach

Stream 提供了新的方法 ‘forEach’ 来迭代流中的每个数据。

Random random = new Random();
random.ints().limit(10).forEach(System.out::println);
3 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());
4 filter

filter 方法用于通过设置的条件过滤出元素。

List<String>strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl");
// 获取空字符串的数量
int count = strings.stream().filter(string -> string.isEmpty()).count();
5 limit

limit 方法用于获取指定数量的流。

Random random = new Random();
random.ints().limit(10).forEach(System.out::println);
6 sorted

sorted 方法用于对流进行排序。

Random random = new Random();
random.ints().limit(10).sorted().forEach(System.out::println);
7 并行流

parallelStream 是流并行处理程序的代替方法。

List<String> strings = Arrays.asList("abc", "", "bc", "efg", "abcd","", "jkl");
// 获取空字符串的数量
int count = strings.parallelStream().filter(string -> string.isEmpty()).count();
8 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());

String mergedString = strings.stream()
    .filter(string -> !string.isEmpty()).collect(Collectors.joining(", "));
9 统计

一些产生统计结果的收集器也非常有用。它们主要用于int、double、long等基本类型上。

List<Integer> numbers = Arrays.asList(3, 2, 2, 3, 7, 3, 5);
 
IntSummaryStatistics stats = numbers.stream().mapToInt((x) -> x).summaryStatistics();
 
System.out.println("列表中最大的数 : " + stats.getMax());
System.out.println("列表中最小的数 : " + stats.getMin());
System.out.println("所有数之和 : " + stats.getSum());
System.out.println("平均数 : " + stats.getAverage());

8 Optional类

Optional 类是一个可以为null的容器对象。如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象。

Optional 是个容器:它可以保存类型T的值,或者仅仅保存null。

Optional提供很多有用的方法,这样我们就不用显式进行空值检测。Optional 类的引入很好的解决空指针异常。

package test.testJava8;

import java.util.Optional;

/**
 * fileName: OptionalTest
 * description: Optional实例
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 11:14
 */
public class OptionalTest {

    public Integer sum(Optional<Integer> a, Optional<Integer> b) {
        System.out.println("第一个参数值是否存在:" + a.isPresent()); // false
        System.out.println("第二个参数值是否存在:" + b.isPresent()); // true

        // 如果值存在,返回它,否则返回默认值
        Integer value1 = a.orElse(new Integer(0));

        // 获取值,只需要存在,不存在会抛出异常
        Integer value2 = b.get();

        return value1 + value2;
    }


    public static void main(String[] args) {
        OptionalTest test = new OptionalTest();

        Integer value1 = null;
        Integer value2 = new Integer(10);

        // 如果为非空,返回 Optional 描述的指定值,否则返回空的 Optional。
        Optional<Integer> a = Optional.ofNullable(value1);
        // 返回一个指定非null值的Optional。
        Optional<Integer> b = Optional.of(value2);

        System.out.println(test.sum(a, b)); // 10
    }
}

9 Nashorn JavaScript

从JDK 1.8开始,Nashorn取代Rhino(JDK 1.6, JDK1.7)成为Java的嵌入式JavaScript引擎。Nashorn完全支持ECMAScript 5.1规范以及一些扩展。

10 日期时间API

Java 8 在 java.time 包下提供了很多新的 API。以下为两个比较重要的 API:

  • Local(本地) − 简化了日期时间的处理,没有时区的问题。
  • Zoned(时区) − 通过制定的时区处理日期时间。

新的java.time包涵盖了所有处理日期,时间,日期/时间,时区,时刻(instants),过程(during)与时钟(clock)的操作。

10.1 本地化日期时间API

LocalDate/LocalTime 和 LocalDateTime 类可以在处理时区不是必须的情况。

package test.testJava8;

import java.time.*;

/**
 * fileName: LocalDateTimeTest
 * description: Java8日期时间API实例
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 11:29
 */
public class DateTimeTest {

    /**
     * 测试本地日期时间API
     */
    public void testLocalDateTime() {
        // 获取当前的日期时间
        LocalDateTime currentTime = LocalDateTime.now();
        System.out.println("当前日期和时间:" + currentTime);
        // 当前日期和时间:2019-09-13T21:41:53.246

        // 获取当前日期
        LocalDate date1 = currentTime.toLocalDate();
        System.out.println("当前日期: " + date1);
        // 当前日期: 2019-09-13

        // 获取月
        Month month = currentTime.getMonth();
        // 获取日
        int day = currentTime.getDayOfMonth();
        // 获取秒
        int second = currentTime.getSecond();

        System.out.println("月:" + month + ",日:" + day + ",秒:" + second);
        // 月:SEPTEMBER,日:13,秒:53

        // 指定年份和月份
        LocalDateTime date2 = currentTime.withDayOfMonth(10).withYear(2019);
        System.out.println("指定年份和月份的日期和时间:" + date2);
        // 指定年份和月份的日期和时间:2019-09-10T21:41:53.246

        // 指定年月日的日期
        LocalDate date3 = LocalDate.of(2019, Month.DECEMBER, 12);
        System.out.println("指定年月日的日期" + date3);
        // 指定年月日的日期2019-12-12

        // 指定小时和分钟的时间
        LocalTime time = LocalTime.of(22, 15);
        System.out.println("指定小时和分钟的时间" + time);
        // 指定小时和分钟的时间22:15

        // 解析时间字符串
        LocalTime time1 = LocalTime.parse("20:15:30");
        System.out.println(time1);
        // 20:15:30

    }

    /**
     * 测试时区日期时间API
     */
    public void testZonedDateTime() {
        ZonedDateTime date1 = ZonedDateTime.parse("2019-09-03t10:34:38+05:30[Asia/Shanghai]");
        System.out.println("解析日期时间字符串:" + date1);
        // 解析日期时间字符串:2019-09-03T10:34:38+08:00[Asia/Shanghai]

        // 指定时区
        ZoneId id = ZoneId.of("Europe/Paris");
        System.out.println("时区id:" + id);
        // 时区id:Europe/Paris

        ZoneId currentZone = ZoneId.systemDefault();
        System.out.println("我所在的时区:" + currentZone);
        // 我所在的时区:Asia/Shanghai

    }

    public static void main(String[] args) {
        DateTimeTest test = new DateTimeTest();
        test.testLocalDateTime();
        test.testZonedDateTime();
    }
}

11 Base64

Java 8 内置了 Base64 编码的编码器和解码器。

Base64工具类提供了一套静态方法获取下面三种BASE64编解码器:

  • 基本: 输出被映射到一组字符A-Za-z0-9+/,编码不添加任何行标,输出的解码仅支持A-Za-z0-9+/。
  • URL: 输出映射到一组字符A-Za-z0-9+_,输出是URL和文件。
  • MIME: 输出隐射到MIME友好格式。输出每行不超过76字符,并且使用’\r’并跟随’\n’作为分割。编码输出最后没有行分割。
package test.testJava8;

import java.io.UnsupportedEncodingException;
import java.util.Base64;
import java.util.UUID;

/**
 * fileName: Base64Test
 * description: Java8 Base64
 *
 * @author lihaogn-main
 * @version 1.0
 * @date 2019/9/3 11:47
 */
public class Base64Test {

    public static void main(String[] args) {

        try {
            // 使用基本编码
            String base64EncodingString = Base64.getEncoder()
                					.encodeToString("lihaogn?java8".getBytes("utf-8"));
            System.out.println("Base64 编码字符串 (基本) :" + base64EncodingString);

            // 解码
            byte[] base64DecodeBytes = Base64.getDecoder().decode(base64EncodingString);
            System.out.println("原始字符串:" + new String(base64DecodeBytes, "utf-8"));

            // 使用URL编码
            base64EncodingString = Base64.getUrlEncoder()
                					.encodeToString("runoob?java8".getBytes("utf-8"));
            System.out.println("Base64 编码字符串 (URL) :" + base64EncodingString);

            // 使用MIME编码
            StringBuilder stringBuilder = new StringBuilder();
            for (int i = 0; i < 10; ++i) {
                stringBuilder.append(UUID.randomUUID().toString());
            }

            byte[] mimeBytes = stringBuilder.toString().getBytes("utf-8");
            String mimeEncodedString = Base64.getMimeEncoder().encodeToString(mimeBytes);
            System.out.println("Base64 编码字符串 (MIME) :" + mimeEncodedString);

        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
    }
}

参考文献

[1] 菜鸟教程:https://www.runoob.com/java/java8-new-features.html

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值