Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >高阶函数和Java的Lambda

高阶函数和Java的Lambda

作者头像
fengzhizi715
发布于 2018-08-24 08:12:14
发布于 2018-08-24 08:12:14
1.9K00
代码可运行
举报
运行总次数:0
代码可运行

2017年的第一天,我坐在独墅湖边,写下这篇文章。

独墅湖.jpeg

在数学和计算机科学中,高阶函数是至少满足下列一个条件的函数:

  • 接受一个或多个函数作为输入
  • 输出一个函数

java世界迎来新的一等公民——函数

java 8引入了函数式编程。函数式编程重点在函数,函数变成了Java世界里的一等公民,函数和其他值一样,可以到处被定义,可以作为参数传入另一个函数,也可以作为函数的返回值,返回给调用者。利用这些特性,可以灵活组合已有函数形成新的函数,可以在更高层次上对问题进行抽象。

使用高阶函数之前的求和、求平方和、求立方和的写法:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class TestHighOrderFunction {
 
   public static int identity(int x) {
      return x;
   }

   public static int sum_integers(int a, int b) {
     int sum = 0;
     for (int i = a; i <= b; i++) {
       sum += identity(i);
     }
    return sum;
   }

   public static int square(int x) {
      return x * x;
   }

  public static int sum_square(int a, int b) {
     int sum = 0;
     for (int i = a; i <= b; i++) {
        sum += square(i);
     }
     return sum;
   }

   public static double cube(int x) {
      return x * x * x;
   }

   public static int sum_cubes(int a, int b) {
      int sum = 0;
      for (int i = a; i <= b; i++) {
         sum += cube(i);
      }
      return sum;
   }

    public static void main(String[] a) {

        System.out.println(sum_integers(1, 10)); // return 55
        System.out.println(sum_square(1, 10));  // return 385
        System.out.println(sum_cubes(1, 10));  // return 3025
    }
}

我们发现sum_开头的方法里,代码很类似,三者唯一区别在于

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   sum += identity(i);
   sum += square(i);
   sum += cube(i);

软件工程里有DRY(don't repeat yourself )的准则。我们来看看使用高阶函数怎样优化刚才的这些代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 public interface Function {
     int opera(int a);
 }

 public static void main(String[] a) {
  
    Function identity = x->x;
    Function square = x->x*x;
    Function cube = x -> x*x*x;
    System.out.println(sum(identity, 1,10)); // return 55
    System.out.println(sum(square, 1,10)); // return 385
    System.out.println(sum(cube, 1,10));   // return 3025
 }
 
 public static int sum(Function term, int a, int b) {
  
     int sum = 0;
     for (int i = a; i <= b; i++) {
        sum += term.opera(i);
     }
     return sum;
 }

得到的结果,跟上面的TestHighOrderFunction类中运行的结果是一样的。不过,这里的sum方法中使用了

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   sum += term.opera(i);

取代了原先的代码。term.opera(i)对应的是原先identity(i)、square(i)、cube(i),在这里Function函数被当做参数进行传递。这就是高阶函数的特性。

对于for循环,我们还能用更优雅的方式进行优化,下面使用了递归的方式。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 public interface Function {
     int opera(int a);
 }

 public static void main(String[] a) {
  
      Function identity = x->x;
      Function square = x->x*x;
      Function cube = x -> x*x*x;
      Function inc = x->x+1; // 定义next函数
      System.out.println(sum(identity, 1,inc,10)); // return 55
      System.out.println(sum(square, 1,inc,10)); // return 385
      System.out.println(sum(cube, 1,inc,10));   // return 3025
 }
 
 public static int sum(Function term, int a, Function next, int b) {
  
      if (a>b) {
          return 0;
      } else {
          return term.opera(a) + sum(term, next.opera(a), next, b);
      }
 }

@FunctionalInterface

在java 8之前我们使用Thread,可能是这样的

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   new Thread(new Runnable() {    
        public void run() {        
              System.out.println("test");    
       }
   }).start();

由于Java 8引入了lambda表达式,我们可以这样写

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   new Thread(()->System.out.println("test")).start();

lambda表达式源于lambda演算。

Lambda演算可以被称为最小的通用程序设计语言。它包括一条变换规则(变量替换)和一条函数定义方式,Lambda演算之通用在于,任何一个可计算函数都能用这种形式来表达和求值。因而,它是等价于图灵机的。尽管如此,Lambda演算强调的是变换规则的运用,而非实现它们的具体机器。可以认为这是一种更接近软件而非硬件的方式。

我们点击Runnable的源码时,发现Runnable使用了@FunctionalInterface,这在java 8之前是没有的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
@FunctionalInterface
public interface Runnable {
    /**
     * When an object implementing interface <code>Runnable</code> is used
     * to create a thread, starting the thread causes the object's
     * <code>run</code> method to be called in that separately executing
     * thread.
     * <p>
     * The general contract of the method <code>run</code> is that it may
     * take any action whatsoever.
     *
     * @see     java.lang.Thread#run()
     */
    public abstract void run();
}

@FunctionalInterface是Java 8为函数式接口引入的一个新的注解。表明该接口是函数式接口,它只包含唯一一个抽象方法。任何可以接受一个函数式接口实例的地方,都可以用lambda表达式。

我们再来看一个匿名函数的例子。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
button.setOnClickListener(new Button.OnClickListener(){   
     public void onClick(View v) {        
         Log.i(TAG,"点击button");    
     }
});

我们将匿名函数改成lambda表达式

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
button.setOnClickListener((v)-> Log.i(TAG,,"点击button"));

这样改造的好处在于,lambda对象的创建是通过字节码指令invokedynamic来完成的,减少了类型和实例的创建消耗。而匿名类需要新的对象的创建。

JDK中的函数式接口举例

java.lang.Runnable, java.awt.event.ActionListener, java.util.Comparator, java.util.concurrent.Callable java.util.function包下的接口,如Consumer、Predicate、Supplier等

简化的lambda——方法引用(Method Reference)

lambda已经简化了代码的写法,然而方法引用进一步简化了lambda的写法。 方法引用的使用方式:类名::方法名

类型

使用方式

备注

引用静态方法

ContainingClass::staticMethodName

Integer::valueOf简化了i->Integer.valueOf(i)的写法

引用特定对象的实例方法

containingObject::instanceMethodName

s::toString()简化了()->s.toString()

引用特定类型的任意对象的实例方法

ContainingType::methodName

System.out::println简化了(s)->System.out.println(s),其中System.out表示的是PrintStream对象

引用构造函数

ClassName::new

String::new简化了()->new String()

我们来看一个简单的例子,对User按照name来进行排序,最初我们会这样写。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  User u1 = new User("tony");
  User u2 = new User("cafei");
  User u3 = new User("aaron");
  
  List<User> users = Arrays.asList(u1,u2,u3);
  
  Collections.sort(users, new Comparator<User>(){

   @Override
   public int compare(User u1, User u2) {
    return u1.getName().compareTo(u2.getName());
   }
   
  });

在java 8以后,Comparator增加了一个静态方法comparing(Function<? super T, ? extends U> keyExtractor),我们可以把排序的写法简化成这样:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Collections.sort(users, Comparator.comparing((User u)->u.getName()));

如果使用方法引用,还可以更加简化代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Collections.sort(users,Comparator.comparing(User::getName));

集合中的应用

在java 8中可以使用新增的api Streams来操作集合,Streams是区别于java.io包里的InputStream 和 OutputStream的概念,是对集合功能的增强。如果你曾经了解过Scala、RxJava等函数式编程,那么看了它的语法以后一定会觉得似曾相识。我们来看两段代码,看看它是如何使用的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
List<Integer> list = Arrays.asList(1, 2, 3, 5, 7, 9, 10)
    .stream()
    .filter(i -> i >= 5)
    .collect(Collectors.toList());

System.out.println("list=" + list); // return list=[5, 7, 9, 10]
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  Arrays.asList("tony", "cafei", "aaron")
           .stream()
           .map(str -> str.toUpperCase())
           .forEach(it -> System.out.println(it));

上面的代码还可以使用方法引用的方式:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  Arrays.asList("tony", "cafei", "aaron")
        .stream()
        .map(String::toUpperCase)
        .forEach(System.out::println);

使用这样的链式调用非常cool。而且,map、filter等方法都是高阶函数。

写在最后

lambda是java 8最为重要的特性,lambda表达式并非只是Java的语法糖,而是由编译器和JVM共同配合来实现的。自从使用了lambda以后我感觉再也回不去了。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2017.01.01 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
500款各领域机器学习数据集,总有一个是你要找的
金融 美国劳工部统计局官方发布数据:http://dataju.cn/Dataju/web/datasetInstanceDetail/139 沪深股票除权除息、配股增发全量数据,截止 2016.12.31 http://dataju.cn/Dataju/web/datasetInstanceDetail/344 上证主板日线数据,截止 2017.05.05,原始价、前复权价、后复权价,1260支股票 http://dataju.cn/Dataju/web/datasetInstanceDetail/340
小莹莹
2018/04/20
4.3K0
资源 | 这是一份非常全面的开源数据集,你,真的不想要吗?
选自Medium 作者:Bharath Raj 机器之心编译 参与:高璇、王淑婷 近期,skymind.ai 发布了一份非常全面的开源数据集。内容包括生物识别、自然图像以及深度学习图像等数据集,现机器之心将其整理如下:(内附链接哦~) 最近新增数据集 开源生物识别数据:http://openbiometrics.org/ Google Audioset:扩展了 632 个音频分类样本,并从 YouTube 视频中提取了 2,084,320 个人类标记的 10 秒声音片段。 地址:https://resea
机器之心
2018/09/20
7600
【资源】最好用的 AI 开源数据集 Top 39:计算机视觉、NLP、语音等 6 大类
【新智元导读】本文按计算机视觉、自然语言处理、语音识别、地理空间数据等人工智能的子领域分类,精心整理,每个数据集均附有下载链接,是做 AI 研究不容错过资源。 今天,构造 AI 或机器学习系统比以往任何时候都更加容易。我们有许多开源的最前沿的工具,如 TesorFlow,Torch,Spark 等,也有 AWS、Google Cloud 以及其他云服务提供商提供的大量计算力,这意味着你可以悠哉地一边喝着咖啡一边用 laptop 训练模型。 虽然不算人工智能这列火车的车头,但 AI 革命的幕后英雄是数据——得
新智元
2018/03/27
1.8K0
【资源】最好用的 AI 开源数据集 Top 39:计算机视觉、NLP、语音等 6 大类
8种寻找机器学习数据集的方法 | 附数据集资源
近日,亚马逊AWS高级技术顾问Will Badr介绍了8种寻找机器学习数据集的方法。
量子位
2019/04/24
1.3K0
8种寻找机器学习数据集的方法 | 附数据集资源
[转载] 机器学习数据集统计系列一
作者:宋天龙 链接:https://www.zhihu.com/question/63383992/answer/222718972 来源:知乎
marsggbo
2018/12/12
1.2K0
入门 | 从文本处理到自动驾驶:机器学习最常用的50大免费数据集
Kaggle:一个数据科学竞赛网站,其中包含大量外部贡献的有趣数据集。你可以在它长长的列表中(https://www.kaggle.com/datasets)找到各种小众数据集,从拉面的评分、篮球数据,到西雅图的宠物牌照。
机器之心
2018/07/30
6640
入门 | 从文本处理到自动驾驶:机器学习最常用的50大免费数据集
资源 | 机器学习高质量数据集大合辑
在机器学习中,寻找数据集也是非常重要的一步。质量高或者相关性高的数据集对模型的训练是非常有帮助的。
磐创AI
2018/11/23
6730
23个优秀的机器学习数据集,给智能更好的经验
在这篇文章中,我会分享 23 个优秀的公共数据集,除了介绍数据集和数据示例外,我还会介绍这些数据集各自可以解决哪些问题。
一点人工一点智能
2022/12/27
1.1K0
23个优秀的机器学习数据集,给智能更好的经验
玩机器学习要知道哪些开源数据库?
开发 AI 和机器学习系统从来没有像现在这样方便。 类似于 TensorFlow、Torch 和 Spark 这样的开源工具,在 AI 开发者群体中已是无处不在。再加上亚马逊 AWS、Google Cloud 等云服务带来的海量计算能力,将来使用笔记本电脑来训练 ML 模型或许不再难以想象。 公众对 AI 的遐想,总忽视了数据的角色。但海量被标记、注解过的数据,是当下 AI 革命当之无愧的主要推手之一。业内研究团队和公司机构,均明白“数据民主化”的意义——使任何开发者都能获取高质量的数据来训练、测试模型,是
AI研习社
2018/03/19
9890
玩机器学习要知道哪些开源数据库?
【资源】想进行数据科学项目却没有数据集?25个数据集网站汇总
原作者 Kunal Jain 编译  Mika 本文为 CDA 数据分析师原创作品,转载需授权 前言 如果用一个句子总结学习数据科学的本质,那就是: 学习数据科学的最佳方法就是应用数据科学。 如果你是初学者,那么每完成一个项目你的能力就会大大提高。如果你是有经验的数据科学从业者,那么你应该懂这个道理。 但是,当我向人们给出这个建议时,他们通常会问:我可以在哪里获得练习的数据集呢? 他们没有意识到存在大量开放的数据集可使用。他们没有意识到通过这些项目,能够不断学习,从而促进自己的职业发展。 如果你认为这符合你
CDA数据分析师
2018/02/26
2K0
【资源】想进行数据科学项目却没有数据集?25个数据集网站汇总
250,000辆汽车–用于机器学习的十大免费车辆图像和视频数据集
随着特斯拉自动驾驶汽车的兴起以及谷歌Waymo等项目的兴起,自动驾驶汽车行业似乎每年都在增长。无人驾驶汽车是计算机视觉的一个重要领域,具有众多应用程序,并且具有巨大的获利潜力。
代码医生工作室
2020/03/04
6.9K0
250,000辆汽车–用于机器学习的十大免费车辆图像和视频数据集
共享单车数据集超10万条
共享单车数据集,包括骑行时间、会员骑行时间、会员类型、骑行路线类别、开始时间、结束时间、开始站点、结束站点、经度纬度等等。
机器学习AI算法工程
2021/09/02
2.3K0
共享单车数据集超10万条
满满的干货:机器学习资料(四)
今天给大侠带来机器学习资料(四),第四篇带来Matlab计算机视觉、自然语言处理、通用机器学习、数据分析/数据可视化、NET计算机视觉、演示及脚本的各种库以及各种资料链接推荐,满满的干货,话不多说,上货。
FPGA技术江湖
2020/12/30
5330
【干货】二十五个深度学习相关公开数据集
(选自Analytics Vidhya;作者:Pranav Dar;磐石编译) 目录 介绍 图像处理相关数据集 自然语言处理相关数据集 语音处理相关数据集 Supplement 一.介绍 通常来说,深度学习的关键在于实践。从图像处理到语音识别,每一个细分领域都有着独特的细微差别和解决方法。 然而,你可以从哪里获得这些数据呢?现在大家所看到的大部分研究论文都用的是专有数据集,这些专有数据集又通常不会公开。那么,想实践那些最新的理论方法往往就成了难题。 如果你也遇到了这样的问题,接下来我们会提供了一系列可用
磐创AI
2018/04/24
1.7K0
【干货】二十五个深度学习相关公开数据集
运动想象,脑电情绪等公开数据集汇总
上述EEG公开数据集汇总整理参考Github用户:meagmohit,Github地址:https://github.com/meagmohit/EEG-Datasets
脑机接口社区
2020/09/30
2.2K0
【Github】Data Competition Top Solution: 数据竞赛top解决方案开源整理
推荐一个Github项目:Smilexuhc/Data-Competition-TopSolution
AINLP
2019/09/20
1.5K0
43种机器学习开源数据集(附地址/调用方法)
学习机器学习是一个不断探索和实验的过程,因此,本文将主要介绍常见的开源数据集,便于学习和实验各种机器学习算法!
扬起
2022/06/30
1.3K0
43种机器学习开源数据集(附地址/调用方法)
推荐阅读
相关推荐
500款各领域机器学习数据集,总有一个是你要找的
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验