一、函数式接口

1、概念

仅有一个抽象方法的接口

适用于函数式编程(Lambda表达式)

常见:Runnable、Comparator<>、生产型接口Producer的get方法、消费型接口Consumer的accept方法、Predict接口的test方法

2、格式

加FunctionalInterface注解进行检测

如果不是函数式接口则编译失败

3、使用

编写实现类,对抽象类中的方法进行实现

在另一个类中定义方法,参数为函数式接口的引用(如Runnable runnable),函数体内调用了函数式接口的抽象方法,即runnable.run()

参数传递实现类的对象

如:show(new MyFunInterfaceImpl());

二、函数式编程

1、概念

一种新的编程范式,区别于过程式编程,如sum(a,b,c)

函数式编程会写为:sum(sum(a,b),c),即把运算过程定义为不同的函数

优点:可以实现延迟执行(第二个参数在第一个参数判断之后),避免性能浪费

2、常见的使用

函数式接口作为参数

函数式接口作为返回值

package com.liujinhui.Day1207FunInterface.ThreadTest;
import java.util.Arrays;
import java.util.Comparator;
/*
,如果一个方法的返回值类型是一个函数式接口,
那么就可以直接返回一个Lambda表达式。
当需要通过一 个方法来获取一个 java.util.Comparator 接口类型的对象作为排序器时,就可以调该方法获取。
* */
public class Demo02Comparator {
//定义一个方法,方法的返回值类型使用函数式接口Comparator
public static Comparator<String> getComparator(){
//方法的返回值类型是一个接口,那么可以返回这个接口的匿名内部类
/*
return new Comparator<String>() {
@Override
public int compare(String o1,String o2){
//按照字符串的降序排序
return o2.length()-o1.length();
}
};
*/
//方法的返回值是一个函数式接口,可以返回一个Lambda表达式
/**/ /*return (String o1,String o2)->{
//按照字符串的降序排序
return o2.length()-o1.length();
};*/
//继续优化Lambda表达式
return (o1,o2)->o2.length()-o1.length();
}
public static void main(String[] args) {
//主方法调用:创建一个字符串数组
String[] arr={"aaa","b","cccc","dddddddddddddd"};
//输出排序前的数组[aaa, b, cccc, dddddddddddddd]
System.out.println(Arrays.toString(arr));
//调用Arrays中的sort方法,对字符串数组进行排序
Arrays.sort(arr,getComparator());
//输出排序后的数组[dddddddddddddd, cccc, aaa, b]
System.out.println(Arrays.toString(arr));
}
}

3、实现方式

  • 匿名内部类:Anonymous classes enable you to declare and instantiate a class at the same time. Use them if you need to use a local class only once.

    • 如:int a = new Scanner(System.in).nextInt();
    • 允许任意类使用
  • Lambda表达式:是一个匿名函数,是jdk8的新特性,允许把函数作为一个方法的参数进行传递,即:方法名(()->{});

    • 只允许函数式接口使用
    • 自动根据上下文推断,可以进行简化,即:方法名(()->);--无return、参数类型、括号,要省全省
    • Lambda表达式≈匿名内部类的语法糖(方便但原理不变,如for-each是Iterator的语法糖)
    • 本质:定义new 接口,内部重写接口的抽象方法
    • Lambda的实例方法引用:如list.stream()..filter(Objects::nonNull).forEach(System.out::println);(本质是System类中PrintStream类型的静态类对象out的println方法,即对元素进行打印输出)
  //使用com.google.guava包创建集合
List<String> list =Lists.newArrayList("a","b","c","d");
//1、正常遍历
list.forEach(item->System.out.println(item));
//2、根据条件遍历
list.forEach(item->{
if("b".equals(item)){
System.out.println(item);
}

逐步优化  

package com.liujinhui.Day1207FunInterface.ThreadTest;
/*
java.lang.Runnable 接口就是一个函数式接口,
假设有一个 startThread 方法使用该接口作为参数,
可以使用Lambda进行传参。这种情况其实和 Thread 类的构造方法参数为 Runnable 没有本质区别。
* */
public class Demo01Runnable {
//定义一个方法stratThread,方法的参数使用函数式接口Runnable
public static void startThread(Runnable run){
//开启多线程
new Thread(run).start();
}
public static void main(String[] args) {
//调用startThread方法,方法的参数是一个接口,可以传递接口的匿名内部类
startThread(new Runnable() {
@Override
public void run() {
System.out.println(Thread.currentThread().getName()+"-->"+"线程启动了");
}
});
//使用Lambda表达式优化函数式接口,参数传递Lambda表达式
startThread(()->{
System.out.println(Thread.currentThread().getName()+"-->"+"线程启动了");
});
//lambda的优化
startThread(()->System.out.println(Thread.currentThread().getName()+"-->"+"线程启动了"));
}
}

4、相同点与不同点

  • 相同点:

    • 都属于函数式编程的内容(逐步实现简化)
  • 不同点:
    • 匿名内部类会被保存为Demo$1.class字节码文件,而Lambda不会保存为class字节码文件
    • 匿名内部类可以用于任意类,Lambda表达式只能用于函数式接口

【重难点】函数式接口、函数式编程、匿名内部类、Lambda表达式、语法糖的更多相关文章

  1. Java8 新特性----函数式接口,以及和Lambda表达式的关系

    这里来讲解一下Java8 新特性中的函数式接口, 以及和Lambda 表达式的关系.看到过很多不少介绍Java8特性的文章,都会介绍到函数式接口和lambda表达式,但是都是分别介绍,没有将两者的关系 ...

  2. Java8函数式接口/Lambda表达式/接口默认方法/接口静态方法/接口冲突方法重写/lambda表达式指定泛型类型等

    一:函数式接口 1.函数式接口的概念就是此接口必须有且只能有一个抽象方法,可以通过@FunctionalInterface来显示规定(类似@Override),但是没有此注解的但是只有一个抽象方法的接 ...

  3. Java 函数式编程(Lambda表达式)与Stream API

    1 函数式编程 函数式编程(Functional Programming)是编程范式的一种.最常见的编程范式是命令式编程(Impera Programming),比如面向过程.面向对象编程都属于命令式 ...

  4. Java8函数式编程以及Lambda表达式

    第一章 认识Java8以及函数式编程 尽管距离Java8发布已经过去7.8年的时间,但时至今日仍然有许多公司.项目停留在Java7甚至更早的版本.即使已经开始使用Java8的项目,大多数程序员也仍然采 ...

  5. Java函数式编程和lambda表达式

    为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...

  6. java8函数式接口详解、函数接口详解、lambda表达式匿名函数、方法引用使用含义、函数式接口实例、如何定义函数式接口

    函数式接口详细定义 函数式接口只有一个抽象方法 由于default方法有一个实现,所以他们不是抽象的. 如果一个接口定义了一个抽象方法,而他恰好覆盖了Object的public方法,仍旧不算做接口的抽 ...

  7. Java 函数式编程和Lambda表达式

    1.Java 8最重要的新特性 Lambda表达式.接口改进(默认方法)和批数据处理. 2.函数式编程 本质上来说,编程关注两个维度:数据和数据上的操作. 面向对象的编程泛型强调让操作围绕数据,这样可 ...

  8. Java8初体验(一)lambda表达式语法

    感谢同事[天锦]的投稿.投稿请联系 tengfei@ifeve.com 本文主要记录自己学习Java8的历程,方便大家一起探讨和自己的备忘.因为本人也是刚刚开始学习Java8,所以文中肯定有错误和理解 ...

  9. Java8 lambda表达式语法 1

    本文主要记录自己学习Java8的历程,方便大家一起探讨和自己的备忘.因为本人也是刚刚开始学习Java8,所以文中肯定有错误和理解偏差的地方,希望大家帮忙指出,我会持续修改和优化.本文是该系列的第一篇, ...

  10. Java8初体验(1):lambda表达式语法

    原文出处: 一冰_天锦 本文主要记录自己学习Java8的历程,方便大家一起探讨和自己的备忘.因为本人也是刚刚开始学习Java8,所以文中肯定有错误和理解偏差的地方,希望大家帮忙指出,我会持续修改和优化 ...

随机推荐

  1. CVE-2022-39197(CobaltStrike XSS <=4.7)漏洞复现

    最新文章更新见个人博客 漏洞说明 根据9.20日CobaltStrike官方发布的最新4.7.1版本的更新日志中介绍,<=4.7的teamserver版本存在XSS漏洞,从而可以造成RCE远程代 ...

  2. ProxySQL(9):ProxySQL的查询缓存功能

    文章转载自: https://www.cnblogs.com/f-ck-need-u/p/9314459.html ProxySQL支持查询缓存的功能,可以将后端返回的结果集缓存在自己的内存中,在某查 ...

  3. Fluentd采集示例

    Fluentd通过读取配置文件来加载各插件,日志经由各插件的处理完成输入到输出的整个路由. 本文通过一个最简单的示例来说明配置文件的结构.td-agent.conf默认位于/etc/td-agent/ ...

  4. 3_JavaScript

    一. 介绍 JavaScript语言诞生主要是完成页面的数据验证, 因此它运行在客户端, 需要运行浏览器来解析执行JavaScript代码 JS是弱类型, Java是强类型 特点 交互性(它可以做的就 ...

  5. NSIS Studio2.1汉化版

    这个东西早些时候是我从老外那里下载回来后放在了群共享里面,中间跟着作者的节奏更新了几次,后来和LmTec聊这个软件的时候提出了汉化的设想,可能是LmTec兄弟看这个软件确实不错,就一口答应了下来,结果 ...

  6. 如何优雅的备份MySQL数据?看这篇文章就够了

    大家好,我是一灯,今天一块学习一下如何优雅安全的备份MySQL数据? 1. 为什么要备份数据 先说一下为什么需要备份MySQL数据? 一句话总结就是:为了保证数据的安全性. 如果我们把数据只存储在一个 ...

  7. zabbix企业监控

    第一节.系统初始化 1.前期环境 主机名 IP地址 操作系统 备注 zabbix-10 192.168.2.10 CentOS Linux release 7.4 zabbix服务端 agent-15 ...

  8. 自建流媒体如何录制视频。齐博x1齐博x2齐博x3齐博x4齐博x5齐博x6齐博x7齐博x8齐博x9齐博x10

    http://x1.eapis.site/ 先打开配置文件\conf\config.php 里边的内容大概如下,第一项是必须要配置的,换成你的网站域名网址.第二项,如果流媒体服务器配置了https证书 ...

  9. 齐博x1标签实例:调用多个圈子同时调用贴子

    下面讲解,在首页,如何调用圈子的同时也调用他们相关的贴子. 单单调用圈子,就像调用文章一样,很多人都能轻松实现,比如下面的代码 {qb:tag name="xxx" type=&q ...

  10. 二、Celery执行一步任务

    二.Celery执行异步任务 2.1.基本使用 创建项目celerypro 创建异步任务执行文件celery_task: import celery import time backend='redi ...