Java 8函数编程轻松入门(四)方法引用
C#中系统提供了许多IEnumerable的扩展方法。同样在Java 8中新引入了Collector类。
1.方法引用
- 定义: 简而言之:就是一个Lambda表达式。在Java 8中,我们我们会使用Lambda表达式创建匿名方法。但是更多的时候,我们的Lambda表达式可能需要调用一个已存在的方法。这时候我们可以通过个一个方法名字来引用已存在的方法会更加清晰。
- 语法: Classname::methodName
- 与C#的类比: 类似C#的委托。C#的委托其实就是方法的指针。而在Java 8中,方法引用其实就是将方法的地址引用指向调用方。
- C#代码
class Program
{
static void Main(string[] args)
{
Func<int, int, int> calc = Add;
}
public static int Add(int i, int j)
{
return i + j;
}
}
- Java 8代码 Lambda表达式
public class Chapter05 {
public static void main(String[] args) {
BinaryOperator<Integer> r = (x, y) -> x + y;
int result = r.apply(1, 2);
}
}
- Java 8代码 方法引用
public class Chapter05 {
public static void main(String[] args) {
BinaryOperator<Integer> r = Chapter05::add;
int result = r.apply(1, 2);
}
public static Integer add(Integer i, Integer j) {
return i + j;
}
}
2.使用Collectors
- 定义: 一种通用的、从流(Stream)生成复杂值的结构。
2.1转换成其他集合(toList(),toSet())
public static void main(String[] args) {
List<Integer> arrays = new ArrayList<Integer>() {{
add(1);
add(2);
add(3);
add(4);
add(5);
}};
List<Integer> afterArrays = arrays.stream().filter(u -> u > 2).collect(Collectors.toList());
}
- List<> 数据结构转换成Set<>数据结构。下面的ArrayList结构转换为HashSet结构
public static void main(String[] args) {
List<Integer> arrays = new ArrayList<Integer>() {{
add(1);
add(2);
add(3);
add(4);
add(5);
}};
Set<Integer> setArrays = arrays.stream().collect(Collectors.toSet());
}
- 上面是Stream类库在背后自动为你挑选合适的数据类型。当然你也可以指定对应类型的数据结构。如下指定TreeSet类型
Set<Integer> treesetArrays=arrays.stream().collect(Collectors.toCollection(TreeSet::new));
2.2转换成值
- 如Stream提供的min\max以及Collectors提供的minBy\maxBy
int max=treesetArrays.stream().max(Comparator.comparing(u->u)).get();
int collectMax=treesetArrays.stream().collect(Collectors.maxBy(Comparator.comparing(u->u))).get();
System.out.println("max:"+max);
System.out.println("CollectMax:"+collectMax);
int avg= treesetArrays.stream().collect(Collectors.averagingInt(u->u)).intValue();
System.out.println("Avg:"+avg);
2.3数据分块
- 类似C#的IEnumerable的GroupBy()
- 有Person的集合。按照性别分组
public class Person {
public String name;
public int age;
public String sex;
public Person(){
}
public Person(String n,int a,String sex){
this.name=n;
this.age=a;
this.sex=sex;
}
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 String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
}
public class GroupByTest {
public static void main(String[] args){
List<Person> persons=new ArrayList<>();
persons.add(new Person("jack",23,"男"));
persons.add(new Person("Tom",20,"男"));
persons.add(new Person("Su",21,"女"));
persons.add(new Person("Lily",22,"女"));
Map<String,List<Person>> group=persons.stream().collect(Collectors.groupingBy(u->u.getSex()));
}
}
2.4Collectors.joining
- 收集流中的数据,生成自定义的格式的字符串
String str=persons.stream().map(u->u.getName()).collect(Collectors.joining(","));
//jack,Tom,Su,Lily
3.Lambda细节
Java 8 可以使用computeIfAbsent缓存。
未使用computeIfAbsent
public class DetailsTest {
public static void main(String[] args){
Map<String,String> keyval=new HashMap<>();
String val=keyval.get("hello");
if(val==null){
keyval.put("hello","world");
}
System.out.print(keyval.get("hello"));
}
}
- 使用computeIfAbsent。现在只需要一句话
public class DetailsTest {
public static void main(String[] args){
Map<String,String> keyval=new HashMap<>();
keyval.computeIfAbsent("hello",u->keyval.put(u,"world"));
}
}
- Map也添加了forEach函数
keyval.forEach((k,v)->{System.out.println("key:"+k+",value:"+v);});
总结:
- 1.方法引用是一种引用方法的轻量级语法,类似C#的委托
- 2.收集器类似C#的IEnumerable的扩展方法
Java 8函数编程轻松入门(四)方法引用的更多相关文章
- Java 8函数编程轻松入门(三)默认方法详解(default function)
default出现的原因 Java 8中对API最大的改变在于集合类,Java在持续演进,但是它一直保持着向后兼容. 在Java 8中为Collection接口增加了stream方法,这意味着所有实现 ...
- Java 8函数编程轻松入门(二)Stream的使用
在C#中,微软基于IEnumerable接口,提供许多便捷的扩展方法,便于实际的开发.在Java 1.8中,Collection接口新增了default stream方法.我们可以针对java集合,在 ...
- Java 8函数编程轻松入门
函数接口介绍 在Java1.8中,新增了Lambda表达式.在.net3.5(C# 3.0)在原先的匿名方法基础上演变出了Lambda表达式.学过C# Lambda表达式的同学,对于Java的Lamb ...
- Java 8函数编程轻松入门(五)并行化(parallel)
1.并发与并行的区别 并发: 一个时间段内有几个程序都处于已启动到运行完毕之间,且这几个程序都是在同一个处理机上运行.但在任一个时刻点只有一个程序在处理机上运行 并行: 在同一个时刻,多核处理多个任务 ...
- Java使用Protocol Buffers入门四步骤
Protocol Buffers(简称protobuf)是谷歌的一项技术.用于将结构化的数据序列化.反序列化.经经常使用于网络传输. 这货实际上相似于XML生成和解析.但protobuf的效率高于XM ...
- Java 8方法引用使用指南
[编者按]本文作者为拥有15年 Java 开发经验的资深程序员 Per-Åke Minborg,主要介绍如何灵活地解析 Java 中的方法引用.文章系国内 ITOM 管理平台 OneAPM 编译呈现. ...
- Java基础教程——方法引用
方法引用 Lambda表达式的代码,是否可以再简洁?--方法引用 对象/类名::方法名 参数都不用写明. import java.util.function.Consumer; public clas ...
- C#基础入门 四
C#基础入门 四 方法参数 值参数:不附加任何修饰符: 输出参数:以out修饰符声明,可以返回一个或多个给调用者: 如果想要一个方法返回多个值,可以用输出参数来处理,输出参数由out关键字标识,如st ...
- Java 8 中的方法引用,轻松减少代码量,提升可读性!
1. 引言 Java8中最受广大开发中喜欢的变化之一是因为引入了 lambda 表达式,因为这些表达式允许我们放弃匿名类,从而大大减少了样板代码,并提高了可读性. 方法引用是lambda表达式的一种特 ...
随机推荐
- 从源码看Azkaban作业流下发过程
上一篇零散地罗列了看源码时记录的一些类的信息,这篇完整介绍一个作业流在Azkaban中的执行过程,希望可以帮助刚刚接手Azkaban相关工作的开发.测试. 一.Azkaban简介 Azkaban作为开 ...
- 高性能IO模型浅析
高性能IO模型浅析 服务器端编程经常需要构造高性能的IO模型,常见的IO模型有四种: (1)同步阻塞IO(Blocking IO):即传统的IO模型. (2)同步非阻塞IO(Non-blocking ...
- transtion:过渡动画
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 17.0px Monaco; color: #4f5d66 } p.p2 { margin: 0.0px 0 ...
- Python学习--04条件控制与循环结构
Python学习--04条件控制与循环结构 条件控制 在Python程序中,用if语句实现条件控制. 语法格式: if <条件判断1>: <执行1> elif <条件判断 ...
- 如何定位Oracle数据库被锁阻塞会话的根源
首先再次明确下,数据库因为要同时保证数据的并发性和一致性,所以操作有锁等待是正常的. 只有那些长时间没有提交或回滚的事物,阻塞了其他业务正常操作,才是需要去定位处理的. 1.单实例环境 2.RAC环境 ...
- 【深入Java虚拟机】之四:类加载机制
类加载过程 类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载.验证.准备.解析.初始化.使用和卸载七个阶段.它们开始的顺序如下图所示: 其中类加载的过程包括了加载.验 ...
- javaScript中的小细节-script标签中的预解析
首先介绍预解析,虽然预解析字面意思很好理解,但是却是出坑出的最多的地方,也是bug经常会有的地方,利用好预解析的特性可以解决很多问题,并且提高代码的质量及数量,浏览器在解析代码前会把变量的声明和函数( ...
- 打破陈规抓痛点,H3 BPM10.0挑战不可能
高效益意味着相似的运营活动比竞争对手做得更好,而战略定位则意味着企业在运营活动中有区别于竞争对手的实施方式,即差异化竞争.在新经济体下,面对社会的变革.市场的竞争环境.不断攀升的成本压力,几乎没有企业 ...
- windows下mongodb配置
打开cmd(windows键+r输入cmd)命令行,进入D:\mongodb\bin目录(如图先输入d:进入d盘然后输入cd d:\mongodb\bin), 输入如下的命令启动mongodb服务: ...
- php安装threads多线程扩展
php5.3或以上,且为线程安全版本.apache和php使用的编译器必须一致.通过phpinfo()查看Thread Safety为enabled则为线程安全版.通过phpinfo()查看Compi ...