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)或事实上最终的
- 实例变量和局部变量背后的实现有一 个关键不同。实例变量都存储在堆中,局部变量则保存在栈上。如果 Lambda 可以直接访问局部变 量,而且 Lambda 是在一个线程中使用的,则使用 Lambda 的线程,可能会在分配该变量的线程将 这个变量收回之后,去访问该变量。因此,Java 在访问自由局部变量时,实际上是在访问它的副本, 而不是访问基本变量。如果局部变量仅仅赋值一次那就没有什么区别了——因此就有了这个限制。
- 这一限制不鼓励你使用改变外部变量的典型命令式编程模式