java各版本的新特性

java各版本的新特性

Java8 新特性

Java 8(又称为jdk 1.8) 是Java语言开发的一个主要版本。Java 8是oracle公司于2014年3月发布, 可以看成是自Java5以来最具革命性的版本。Java 8为Java语言、编译器、类库、开发工具与JVM带来了大量新特性。

1. Lambda表达式

为什么使用Lambda表达式
Lambda是一个匿名函数, 我们可以把Lambda表达式理解为是一段可以传递的代码(将代码像数据一样进行传递)。使用它可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格, 使Java的语言表达能力得到了提升。

在Java 8中, Lambda表达式是对象, 而不是函数, 它们必须依附于一类特别的对象类型:函数式接口。

语法格式一:无参,无返回值
@Test
public void test() {
   
    Runnable r1 = new Runnable() {
   
        @Override
        public void run() {
   
      	   System.out.println("lalalalalalala");
        }
    };
    r1.run();
    
    Runnable r2=()->System.out.println("lalalalalalala");
    r2.run();
}
语法格式二:Lambda需要一个参数, 但是没有返回值
@Test
public void test() {
   
    Consumer<String> con = new Consumer<String>() {
   
        @Override
        public void accept(String s) {
   
        	System.out.println(s);
        }
    );
    con.accept(t:"lalalalalalal");
  
    Consumer<String> con1 = (String s) ->{
   
    	System.out.println(s);
    };
    con1.accept(t:"lalalalalalalala");
}
语法格式三:类型推断

数据类型可以省略,因为可由编译器推断得出,称为“类型推断”。

@Test
public void test() {
   
    Consumer<String> con1 = (String s) ->{
   
   	 	System.out.println(s);
    };
    con1.accept(t:"lalalalalalaala");        
 
    Consumer<String> con2 = (s) ->{
   
    	System.out.println(s);
    };
    con2.accept(t:"lalalalalalaaaala");
}
语法格式四:Lambda若只需要一个参数时, 参数的小括号可以省略
@Test
public void test() {
   
    Consumer<String> con1 = (s) ->{
   
    	System.out.println(s);
    };
    con1.accept(t:"lalalallalala");
   
    Consumer<String> con2 = s->{
   
   	 	System.out. println(s);
    };
    con2.accept(t:"lalalalalalalalala");
}
语法格式五:Lambda需要两个或以上的参数, 多条执行语句, 并且可以有返回值
@Test
public void test() {
   
    Comparator<Integer> com1 = new Comparator<Integer>() {
   
        @override
        public int compare(Integer o1,Integer o2) {
   
            System.out.println(o1);
            System.out.println(o2);
            return o1.compareTo(o2);
        }
    };
    System.out.println(com1.compare(12,21));

    Comparator<Integer> com2 = (o1,o2) ->{
   
        System.out.println(o1);
        System.out.println(o2);
        return o1.compareTo(o2);
    };
    System.out.println(com2.compare(12,6));
}
语法格式六:当Lambda体只有一条语句时, return与大括号若有, 都可以省略
@Test
public void test() {
   
    Comparator<Integer> com1 = (o1,o2) ->{
   
    	return o1.compareTo(o2);
    };
    System.out.println(com1.compare(12,6));
  
    Comparator<Integer> com2 = (o1,o2) ->o1.compareTo(o2);
    System.out.println(com2.compare(12,21));
}

2. 函数式(Functional) 接口

什么是函数式接口

  • 只包含一个抽象方法的接口,称为函数式接口。
  • 你可以通过Lambda表达式来创建该接口的对象。(若Lambda表达式抛出一个受检异常(即:非运行时异常),那么该异常需要在目标接口的抽象方法上进行声明)。
  • 我们可以在一个接口上使用@Functionallnterface注解, 这样做可以检查它是否是一个函数式接口。同时javadoc也会包含一条声明, 说明这个接口是一个函数式接口。
  • 在java.util.function包下定义了Java 8的丰富的函数式接口。

如何理解函数式接口

  • Java从诞生日起就是一直倡导“一切皆对象”, 在Java里面面向对象(OOP)编程是一切。但是随着python、scala等语言的兴起和新技术的挑战, Java不得不做出调整以便支持更加广泛的技术要求, 也即java不但可以支持OOP还可以支持O OF(面向函数编程)。
  • 在函数式编程语言当中,函数被当做一等公民对待。在将函数作为一等公民的编程语言中, Lambda表达式的类型是函数。但是在Java 8中, 有所不同。在Java 8中, Lambda表达式是对象, 而不是函数, 它们必须依附于一类特别的对象类型——函数式接口。
  • 简单的说, 在Java 8中, Lambda表达式就是一个函数式接口的实例。这就是Lambda表达式和函数式接口的关系。也就是说, 只要一个对象是函数式接口的实例, 那么该对象就可以用Lambda表达式来表示。
  • 所以以前用匿名实现类表示的现在都可以用Lambda表达式来写。

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

函数式接口 参数类型 返回类型 用途
Consumer 消费型接口 T void 对类型为T的对象应用操作,包含方法:
void accept(T t)
Supplier 供给型接口 T 返回类型为T的对象, 包含方法:T get()
Function<T,R> 函数型接口 T R 对类型为T的对象应用操作,
并返回结果。结果是R类型的对象。
包含方法:R apply(T t)
Predicate 断定型接口 T boolean 确定类型为T的对象是否满足某约束,
并返回boolean值。包含方法:boolean test(T t)

3. 方法引用与构造器引用

方法引用(Method References)
  • 当要传递给Lambda体的操作, 已经有实现的方法了, 可以使用方法引用!
  • 方法引用可以看做是Lambda表达式深层次的表达。换句话说, 方法引用就是Lambda表达式, 也就是函数式接口的一个实例, 通过方法的名字来指向一个方法, 可以认为是Lambda表达式的一个语法糖。
  • 要求:实现接口的抽象方法的参数列表和返回值类型,必须与方法引用的方法的参数列表和返回值类型保持一致!
  • 格式:使用操作符“::”将类(或对象)与方法名分隔开来。
  • 如下三种主要使用情况:

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

构造器引用(Constructor References)

SuppLier中的 T get()
EmpLoyee的空参构造器:EmpLoyee()

@Test
public void test() {
   
    Supplier<Employee> sup = new Supplier<Employee>() {
   
        @Override
        public Employee get() {
   
           return new Employee();
        }
    };
    
    System.out.println("*****************");
    
    Supplier<Employee> sup1 = () -> new Employee();
    System.out.println(sup1.get());
    
    System.out.println("******************");
    
    Supplier<Employee> sup2 = Employee::new;
    System.out.println(sup2.get());
}

4. 强大的Stream API

什么是Stream
  • Stream到底是什么呢?是数据渠道,用于操作数据源(集合、数组等)所生成的元素序列。“集合讲的是数据, Stream讲的是计算!
  • 注意
    ①Stream自己不会存储元素。
    ②Stream不会改变源对象。相反, 他们会返回一个持有结果的新Stream。
    ③Stream操作是延迟执行的。这意味着他们会等到需要结果的时候才执行。
Stream API说明
  • Java 8中有两大最为重要的改变。第一个是Lambda表达式; 另外一个则是Stream API。
  • Stream API(java.util stream) 把真正的函数式编程风格引入到Java中。这是目前为止对Java类库最好的补充, 因为Stream API可以极大提供Java程序员的生产力,让程序员写出高效率、干净、简洁的代码。
  • Stream是Java 8中处理集合的关键抽象概念, 它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。使用Stream API对集合数据进行操作, 就类似于使用SQL执行的数据库查询。也可以使用Stream API来并行执行操作。简言之, Stream API提供了一种高效且易于使用的处理数据的方式。
  • Java 8可以透明地把输入的不相关部分拿到几个CPU内核上去分别执行你的Stream操作流水线。
  • Stream支持许多处理数据的并行操作,其思路和在数据库查询语言中的思路类似。
为什么要使用Stream API
  • 实际开发中, 项目中多数数据源都来自于Mysql, Oracle等。但现在数据源可以更多了, 有Mon gDB, Red is 等, 而这些No SQL的数据就需要Java层面去处理。
  • Stream和Collection集合的区别:Collection是一种静态的内存数据结构, 而Stream是有关计算的。前者是主要面向内存, 存储在内存中,后者主要是面向CPU, 通过CPU实现计算。
  • 例如你一个接口如果里面不仅要返回全部数据,还要返回满足某个条件的数据,用stream处理结果集就可以少操作一次数据库了,这样反而可能提高了性能,同时代码也简洁了。
并行流与串行流

并行流就是把一个内容分成多个数据块,并用不同的线程分别处理每个数据块的流。相比较串行的流,并行的流可以很大程度上提高程序的执行效率。Java 8中将并行进行了优化, 我们可以很容易的对数据进行并行操作。Stream API 可以声明性地通过parallel() 与sequential() 在并行流与顺序流之间进行切换。

Stream的操作三个步骤
创建Stream

一个数据源(如:集合、数组),获取一个流。

创建Stream方式一:通过集合

Java 8中的Collection接口被扩展, 提供了两个获取流的方法:

  • default Stream stream() :返回一个顺序流

  • default Stream parallelStream() :返回一个并行流

@Test
public void test() {
    
 List<Employee> employees = EmployeeData.getEmpLoyees();
 //defauLt Stream<E> stream() :返回一个顺序流
 Stream<Employee> stream = employees.stream();

 //default Stream<E>paralLel Stream() :返回一个并行流
 Stream<Employee> parallelStream = employees.parallelStream();
}

创建Stream方式二:通过数组
Java 8中的Arrays的静态方法stream() 可以获取数组流:

static Stream stream(T[] array) :返回一个流

重载形式,能够处理对应基本类型的数组:

  • public static Int Stream stream(int[] array)
  • public static Long Stream stream(long[] array)
  • public static Double Stream stream(double[] array)
@Test
public void test() {
    
 int[] arr=new int[] {
    1,2,3,4,5,6};
 IntStream stream = Arrays.stream(arr);
 
 Employee e1 = new Employee (id: 1001, name:"Tom");
 Employee e2 = new Employee (id: 1002, name:"Jerry");    
 Employee[] arr1 = new Employee[]{
    e1, e2};
 
 Stream<Employee> stream1 = Arrays.stream(arr1);
}

创建Stream方式三:通过Stream的of()

可以调用Stream类静态方法of() , 通过显示值创建一个流。它可以接收任意数量的参数。
public static Stream of(T…values) :返回一个流

@Test
public void test() {
   
	Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5, 6);
}

创建Stream方式四:创建无限流

  • 4
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值