Java8新特性_Stream_筛选与切片

这篇博客通过一系列测试用例详细展示了Java Stream API的使用,包括过滤、映射、短路操作、去重等。主要涉及员工对象列表的处理,如筛选年龄大于35的员工、打印姓名、去重等操作,强调了Stream API的内部迭代特性及其在效率上的优势。

package com.zs.boot.controller;

public class Employee {

	private int id;
	private String name;
	private int age;
	private double salary;
	private Status status;

	public Employee() {
	}

	public Employee(String name) {
		this.name = name;
	}

	public Employee(String name, int age) {
		this.name = name;
		this.age = age;
	}

	public Employee(int id, String name, int age, double salary) {
		this.id = id;
		this.name = name;
		this.age = age;
		this.salary = salary;
	}

	public Employee(int id, String name, int age, double salary, Status status) {
		this.id = id;
		this.name = name;
		this.age = age;
		this.salary = salary;
		this.status = status;
	}

	public Employee(String name, int age, double salary) {
		this.name = name;
		this.age = age;
		this.salary = salary;
	}

	public Status getStatus() {
		return status;
	}

	public void setStatus(Status status) {
		this.status = status;
	}

	public int getId() {
		return id;
	}

	public void setId(int id) {
		this.id = id;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}

	public double getSalary() {
		return salary;
	}

	public void setSalary(double salary) {
		this.salary = salary;
	}

	public String show() {
		return "测试方法引用!";
	}

	@Override
	public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + age;
		result = prime * result + id;
		result = prime * result + ((name == null) ? 0 : name.hashCode());
		long temp;
		temp = Double.doubleToLongBits(salary);
		result = prime * result + (int) (temp ^ (temp >>> 32));
		return result;
	}

	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		Employee other = (Employee) obj;
		if (age != other.age)
			return false;
		if (id != other.id)
			return false;
		if (name == null) {
			if (other.name != null)
				return false;
		} else if (!name.equals(other.name))
			return false;
		if (Double.doubleToLongBits(salary) != Double.doubleToLongBits(other.salary))
			return false;
		return true;
	}

	@Override
	public String toString() {
		return "Employee [id=" + id + ", name=" + name + ", age=" + age + ", salary=" + salary + ", status=" + status
				+ "]";
	}

	public enum Status {
		FREE, BUSY, VOCATION;
	}

}
package com.zs.boot.controller;

import org.junit.Test;

import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.stream.Stream;

public class TestStream2API {

    List<Employee> employees = Arrays.asList(
            new Employee("刘德华",48,8000),
            new Employee("张学友",28,8500),
            new Employee("黎明",45,6500),
            new Employee("黎明",45,6500),
            new Employee("郭富城",51,5500));

    //过滤出年龄大于35
    //内部迭代:迭代操作由StreamAPI完成
    @Test
    public void test1(){
        //中间操作,不会执行任何操作
        Stream<Employee> s = employees.stream().filter(e -> e.getAge()>35);

        /*
        s.forEach(System.out::println);这样写会打印对象所有信息,不会只打印姓名,而我只打印姓名
        可以参考这样写Stream<String> namesStream = employees.stream().map(Employee::getName);
        Stream<String> namesStream  =  s.map(employees -> employees.getName());//用Employee::getName更简洁
        namesStream.forEach(System.out::println);
        */

        //中间操作后必须加终止操作才执行
        s.forEach((str) -> System.out.println(str.getName()));
        /*
        刘德华
        黎明
        郭富城
        */

    }

    //外部迭代
    @Test
    public void test2(){
        Iterator<Employee> it = employees.iterator();
        while (it.hasNext()){
            System.out.println(it.next());
        }
    }

    @Test
    public void test3(){
        /*Employee [id=0, name=刘德华, age=48, salary=8000.0, status=null]
        Employee [id=0, name=张学友, age=28, salary=8500.0, status=null]
        Employee [id=0, name=黎明, age=45, salary=6500.0, status=null]*/
        employees.stream().filter(e -> e.getSalary()>6000).limit(3).forEach(System.out::println);

        System.out.println("------------------------------------------------------");

        /*短路!
        Employee [id=0, name=刘德华, age=48, salary=8000.0, status=null]
        短路!
        Employee [id=0, name=张学友, age=28, salary=8500.0, status=null]*/
        //找到满足条件的两条数据后,后续的迭代操作就不再进行了
        employees.stream().filter(e -> {
            System.out.println("短路!");
            return e.getSalary()>6000;
        }).limit(2).forEach(System.out::println);
    }

    /*Employee [id=0, name=黎明, age=45, salary=6500.0, status=null]*/
    //跳过前两个
    @Test
    public void test4(){
        employees.stream().filter(e -> e.getSalary()>6000).skip(2).forEach(System.out::println);
    }

    /*Employee [id=0, name=刘德华, age=48, salary=8000.0, status=null]
    Employee [id=0, name=张学友, age=28, salary=8500.0, status=null]
    Employee [id=0, name=黎明, age=45, salary=6500.0, status=null]*/
    //去重,注意要去实体类中重写hashcode和equals
    @Test
    public void test5(){
        employees.stream().filter(e -> e.getSalary()>6000).distinct().forEach(System.out::println);
    }

}

Java 8新特性众多,涵盖了语法、API和工具等多个方面,为开发者提供了更高效、便捷的编程体验。 ### 语法层面 - **Lambda表达式**:是一种简洁的语法结构,用来表示匿名函数。其基本语法为`(参数列表) -> { 代码块 }`,如`(a, b) -> a + b`。它能让代码更简洁,提升开发效率。Lambda表达式可访问外层或effectively final(隐式final)的局部变量,实例变量和静态变量可直接访问且无限制。JVM通过invokedynamic指令动态绑定,生成匿名内部类,每个Lambda表达式在首次调用时生成一个类 [^3]。 - **方法引用**:提供了一种更简洁的方式来引用已存在的方法,包括对象::实例方法名、类::静态方法名、类::实例方法名等形式 [^4]。 - **默认方法**:允许在接口中定义具有默认实现的方法,这样实现该接口的类可以不实现这些方法。这一特性有助于在不破坏现有实现类的情况下,向接口中添加新方法 [^1][^2][^4]。 - **静态方法**:接口中可以定义静态方法,可直接通过接口名调用。例如: ```java interface MathUtils { static int add(int a, int b) { return a + b; } } // 调用接口中的静态方法 int sum = MathUtils.add(3, 5); System.out.println(sum); // 输出: 8 ``` - **新的重复注解**:允许在同一个地方多次使用相同的注解,增强了注解的灵活性 [^5]。 ### API层面 - **Stream API**:用于处理集合数据,提供了创建流、中间操作(如筛选切片、映射、排序等)和终止操作(如查找、匹配、归约、收集等)等功能,使代码更简洁、易读,提高了集合数据处理的效率 [^2][^4]。 - **Date Time API**:提供了新的日期和时间处理类,如`LocalDate`、`LocalTime`、`LocalDateTime`等,解决了旧日期时间API的线程安全问题,并且提供了更丰富的操作方法 [^1]。 - **Optional类**:用于解决空指针问题,通过封装可能为null的对象,提供了一些方法来安全地处理空值情况 [^2][^4]。 ### 工具层面 Java 8引入了新编译工具,提升了编译效率和性能 [^1]。 ### 其他层面 - **Nashorn javascript引擎**:允许在JVM上运行JavaScript代码,增强了Java的动态脚本执行能力 [^1]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ZHOU_VIP

您的鼓励将是我创作最大的动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值