JAVA 8 实战书籍记录

常用API

操 作 类 型 返回类型 使用的类型/函数式接口 函数描述符
filter 过滤 Stream Predicate
distinct 去重 Stream
skip 跳过流数据 Stream long
limit 截取流数据 Stream long
map 转换map获取特定字段 Stream Function<T, R>
flatMap 每次遍历都是一个元素 Stream Function<T, Stream>
sorted 排序 Stream Comparator
anyMatch 随机匹配 boolean Predicate
noneMatch 匹配没有任何相等的 boolean Predicate
allMatch 所有匹配 boolean Predicate
findAny Optional
findFirst 寻找第一个 Optional
forEach void Consumer
collect R Collector<T, A, R>
reduce 递增 Optional BinaryOperator
count long

实际案例

//Trader类
public class Trader{
	private final String name;
	private final String city;
    
	public Trader(String n, String c){
	this.name = n;
	this.city = c;
	}
    
	public String getName(){
	return this.name;
	}
    
	public String getCity(){
	return this.city;
	}
    
	public String toString(){
	return "Trader:"+this.name + " in " + this.city;
	}  
}
//Transaction类
public class Transaction{
private final Trader trader;
private final int year;
private final int value;
    
	public Transaction(Trader trader, int year, int value){
	this.trader = trader;
	this.year = year;
	this.value = value;
	}
    
	public Trader getTrader(){
	return this.trader;
	}
    
	public int getYear(){
	return this.year;
	}
    
	public int getValue(){
	return this.value;
	}
    
	public String toString(){
	return "{" + this.trader + ", " +
	"year: "+this.year+", " +
	"value:" + this.value +"}";
	}
}
//赋值
Trader raoul = new Trader("Raoul", "Cambridge");
Trader mario = new Trader("Mario","Milan");
Trader alan = new Trader("Alan","Cambridge");
Trader brian = new Trader("Brian","Cambridge");
List<Transaction> transactions = Arrays.asList(
new Transaction(brian, 2011, 300),
new Transaction(raoul, 2012, 1000),
new Transaction(raoul, 2011, 400),
new Transaction(mario, 2012, 710),
new Transaction(mario, 2012, 700),
new Transaction(alan, 2012, 950)
);

找出2011年的所有交易并按交易额排序(从低到高)

List<Transaction> tr2011 =transactions.stream().filter(transaction -> transaction.getYear() == 2011).sorted(comparing(Transaction::getValue)).collect(toList());

交易员都在哪些不同的城市工作过

List<String> cities =transactions.stream().map(transaction -> transaction.getTrader().getCity()).distinct().collect(toList());

可以去掉 distinct() ,改用 toSet() ,这样就会把流转换为集合)

Set<String> cities =transactions.stream().map(transaction -> transaction.getTrader().getCity()).collect(toSet());

查找所有来自于剑桥的交易员,并按姓名排序

List<Trader> traders =transactions.stream().map(Transaction::getTrader).filter(trader -> trader.getCity().equals("Cambridge")).distinct().sorted(comparing(Trader::getName)).collect(toList());

返回所有交易员的姓名字符串,按字母顺序排序

String traderStr =transactions.stream().map(transaction -> transaction.getTrader().getName()).distinct().sorted().reduce("", (n1, n2) -> n1 + n2);

此解决方案效率不高(所有字符串都被反复连接,每次迭代的时候都要建立一个新的 String 对象),使用 joining (其内部会用到 StringBuilder )

String traderStr =transactions.stream().map(transaction -> transaction.getTrader().getName()).distinct().sorted().collect(joining());

有没有交易员是在米兰工作的

boolean milanBased =transactions.stream().anyMatch(transaction -> transaction.getTrader().getCity().equals("Milan"));

打印生活在剑桥的交易员的所有交易额

transactions.stream().filter(t -> "Cambridge".equals(t.getTrader().getCity())).map(Transaction::getValue).forEach(System.out::println);

所有交易中,最高的交易额是多少

Optional<Integer> highestValue =transactions.stream().map(Transaction::getValue).reduce(Integer::max);

找到交易额最小的交易

Optional<Transaction> smallestTransaction =transactions.stream().reduce((t1, t2) ->t1.getValue() < t2.getValue() ? t1 : t2);

流支持 min 和 max 方法,可以接受一个 Comparator 作为参数,指定计算最小或最大值时要比较哪个键值:

Optional<Transaction> smallestTransaction =transactions.stream().min(comparing(Transaction::getValue));

lambda表达式使用局部变量

   int portNumber = 1337;
   Runnable r = () -> System.out.println(portNumber);
   portNumber = 31337;
   ------------------------------------------
//如果要改变变量需要再lambda使用,用一个参数获取未修改前的值,默认是final不用使用修饰符

错误:Lambda 表达式引用 的局部变量必须是最终的 (final)或事实上最终的

  1. 实例变量和局部变量背后的实现有一 个关键不同。实例变量都存储在堆中,局部变量则保存在栈上。如果 Lambda 可以直接访问局部变 量,而且 Lambda 是在一个线程中使用的,则使用 Lambda 的线程,可能会在分配该变量的线程将 这个变量收回之后,去访问该变量。因此,Java 在访问自由局部变量时,实际上是在访问它的副本, 而不是访问基本变量。如果局部变量仅仅赋值一次那就没有什么区别了——因此就有了这个限制。
  2. 这一限制不鼓励你使用改变外部变量的典型命令式编程模式
posted @ 2022-03-29 16:17  Kang_kin  阅读(40)  评论(0)    收藏  举报