前言:

这两天彻底的复习了一遍Java8的各种新特性,趁着热乎劲,把知识点整理成博客的形式保存一下。



### 一、Lambda介绍
**Lambda表达式 :** 也可称为闭包,Lambda允许把函数作为一个方法的参数(函数作为参数传递到方法中),免去了使用匿名方法的麻烦,并且给予Java简单但是强大的函数化的编程能力


语法格式:

(parameters) -> expression

或者

(parameters) ->{ statements; }

格式说明:

():接口中抽象方法的参数列表,没有参数就空着,有参数则写出参数,多个参数用逗号分隔

->:传递的意思,将参数传递给方法体{}

{}:重写接口的抽象方法的方法体

二、Lambda用法实例

1.简单实例(基础用法的规则)

1.1 无参数时,可以省略括号内容
()->10		//返回结果值:10
1.2 传入String类型的参数
(String s)->System.out.print(s)		//打印s的内容
1.3 传入的参数类型可以省略,如下
//
a->a*10 //返回结果值:a*10(传入一个参数时,可以省略括号) (a,b)->a*b //返回结果值:a*b(传入两个或以上参数时,不能省略括号,)

#### 2.函数式接口
##### 2.1 Lambda实现自定义接口Calculator
首先定义一个函数式接口 ` Calculator `,包含唯一一个抽象方法 ` Calcu() `
```
public interface Calculator {
public abstract int Calcu(int x, int y);
}
```


其次定义一个 ` invokeCalcu() `,接收参数类型为接口
```
public static void invokeCalcu(int x, int y, Calculator calculator) {
int sum = calculator.Calcu(x, y);
System.out.println("sum = " + sum);
}
```


最后调用 ` invokeCalcu() `,一共有两种方法实现:
1)使用匿名内部类传入接口,并实现抽象方法
```
invokeCalcu(10, 20, new Calculator() {
@Override
public int Calcu(int x, int y) {
return x + y;
}
});
```
2)使用Lambda表达式,简化操作
```
invokeCalcu(120, 130, (a, b) -> a + b);
```



##### 2.2 Lambda表达式实现**多线程**接口实例 ` Runnable `
```
//使用匿名内部类的方式实现多线程的创建
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("currentThreadName = " + Thread.currentThread().getName());
}
}).start();

//使用lambda表达式实现多线程

new Thread(() -> System.out.println("currentThread().getName() = " + Thread.currentThread().getName())).start();

<br><hr>

##### 2.3 Lambda表达式实现**比较器**接口实例  ` Comparator `
定义一个Person类,保存姓名和年龄

public class Person {

private String name;

private int age;

public Person() { }

public Person(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
} 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; }

}

<br>
然后分别使用匿名内部类和Lambda表达式对Person类的年龄进行排序

Person[] arr = {

new Person("佟丽娅", 26),

new Person("范冰冰", 22),

new Person("柳岩", 21)

};

//匿名内部类重写comparator接口里的compare()

System.out.println("=匿名内部类升序排序:===========");

Arrays.sort(arr, new Comparator() {

@Override
public int compare(Person o1, Person o2) {
return o1.getAge() - o2.getAge();
}

});

for (Person person : arr) {

System.out.println(person);

}

//使用Lambda表达式,简化匿名内部类

System.out.println("=Lambda表达式降序排序:=========");

Arrays.sort(arr, (o1, o2) -> o2.getAge() - o1.getAge());

/**

  • jdk1.8 也可以这样写 方法引用
  • Arrays.sort(arr,Comparator.comparingInt(Person::getAge).reversed());

    **/

    for (Person person : arr) {

    System.out.println(person);

    }

//Stream API 也是Java8的新特新写法

System.out.println("=Stream-sort升序排序:=========");

Arrays.stream(arr).sorted(Comparator.comparing(Person::getAge)).forEach(System.out::println);


<br><hr> > 接口中 **有且仅有唯一一个抽象方法** (接口中可以包含其他私有、默认、静态的方法),称之为函数式接口(这种类型的接口也称为SAM接口,即Single Abstract Method interfaces)
> > **使用前提:**
> > - 使用Lambda必须具有接口,且要求**接口中有且仅有一个抽象方法**
> > - 无论是 JDK内置的 ` Runnable ` 、` Comparator ` 接口还是自定义的接口,只有当接口中的抽象方法存在且唯一时,才可以使用Lambda
> > - 使用Lambda必须具有**上下文推断**
> > - 方法的**参数或局部变量类型**必须为Lambda对应的接口类型,才能使用Lambda作为该接口的实例
> > > 有且仅有一个抽象方法的接口,称为 ` “函数式编程” ` ### 三、Lambda变量作用域
##### 3.1在 Lambda 表达式当中不允许声明一个与局部变量同名的参数或者局部变量

String[] str = {"1","12","123"};

Comparator comparator = (str, second) -> Integer.compare(str.length(), second.length()); //str处的编译会出错

<br><hr>
#####3.2 Lambda 表达式只能引用标记了 final 的外层局部变量
> 也就是说不能在 lambda 内部修改定义在域外的局部变量,否则会编译错误。 可以直接在 lambda 表达式中访问外层的局部变量:

public static void main(String args[]) {

final int num = 1;

Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));

s.convert(2); // 输出结果为 3

}

public interface Converter<T1, T2> {
void convert(int i);
}
<br>
lambda 表达式的局部变量可以不用声明为 final,但是必须不可被后面的代码修改(即隐性的具有 final 的语义)

int num = 1;

Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));

s.convert(2);

num = 5; //由于在Lambda表达式中引用了num,所以num是隐形的final修饰,但是这里修改了num的值,final就不存在了

Java8新特性 (一)Lambda的更多相关文章

  1. java8新特性: lambda表达式:直接获得某个list/array/对象里面的字段集合

    java8新特性: lambda表达式:直接获得某个list/array/对象里面的字段集合 比如,我有一张表: entity Category.java service CategoryServic ...

  2. Java8 新特性学习 Lambda表达式 和 Stream 用法案例

    Java8 新特性学习 Lambda表达式 和 Stream 用法案例 学习参考文章: https://www.cnblogs.com/coprince/p/8692972.html 1.使用lamb ...

  3. Java8新特性系列-Lambda

    转载自:Java8新特性系列-Lambda – 微爱博客 Lambda Expressions in Java 8 Lambda 表达式是 Java 8 最流行的特性.它们将函数式编程概念引入 Jav ...

  4. Java8新特性之Lambda表达式

    lambda表达式是java8给我们带来的几个重量级新特性之一,借用lambda表达式,可以让我们的java程序设计更加简洁.最近新的项目摒弃了1.6的版本,全面基于java8进行开发,本文是java ...

  5. Java8 新特性之Lambda表达式

    1. Lambda 表达式概述 Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递); Lambda 表达式可以写出更简洁,更灵活的代码 ...

  6. 【Java8新特性】Lambda表达式基础语法,都在这儿了!!

    写在前面 前面积极响应读者的需求,写了两篇Java新特性的文章.有小伙伴留言说:感觉Lambda表达式很强大啊!一行代码就能够搞定那么多功能!我想学习下Lambda表达式的语法,可以吗?我的回答是:没 ...

  7. 零基础学习java------21---------动态代理,java8新特性(lambda, stream,DateApi)

    1. 动态代理 在一个方法前后加内容,最简单直观的方法就是直接在代码上加内容(如数据库中的事务),但这样写不够灵活,并且代码可维护性差,所以就需要引入动态代理 1.1 静态代理实现 在讲动态代理之前, ...

  8. 【Java8新特性】- Lambda表达式

    Java8新特性 - Lambda表达式 生命不息,写作不止 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 一个有梦有戏的人 @怒放吧德德 分享学习心得,欢迎指正,大家一起学习成长! ...

  9. 漫漫人生路-学点Jakarta基础-Java8新特性 Stream/Lambda

    背景 Stream 是对集合(Collection)对象功能的增强,它专注于对集合对象进行各种非常便利.高效的聚合操作(aggregate operation),或者大批量数据操作 (bulk dat ...

  10. Java8新特性之Lambda

    为什么要Lambda Java8应该是目前最大的一次更新了,更新后我们迎来了很多新特性,其中便包括Lambda表达式,函数式编程的思想正式进入Java,让我们看一个经典案例. 例1 按照两个人的年龄排 ...

随机推荐

  1. 如何用JS获取地址栏参数的方法

    采用正则表达式获取地址栏参数: 写一个方法来进行正则匹配,同样也可以复用 function GetPar(name) { var reg = new RegExp("(^|&)&qu ...

  2. kali安装结束重启黑屏?

    很多人碰到过kali在安装结束后自动重启,屏幕黑屏就显示一个光标. 解决办法: 安装最后一步,不要选择默认项 Enter device manually 改选第二项.....具体什么忘记了. 即可解决 ...

  3. 巧用浏览器F12调试器定位系统前后端bug-转载

    做测试的小伙伴可能用过httpwatch,firebug,fiddler,charles等抓包(数据包)工具,但实际上除了这些还有一个简单实用并的抓包工具,那就是浏览器的F12调试器. httpwat ...

  4. minikube国内在线部署体验

    问题描述: 快速学习k8s的各个组件的作用及yml的编写,minikube很适合. how to install Minikube, a tool that runs a single-node Ku ...

  5. AD10生成Gerber文件详细步骤

    参考:https://wenku.baidu.com/view/faf0363c195f312b3069a5d2.html

  6. Hibernate的悲观锁和乐观锁

    前一篇博客我们从数据库角度分析,锁可以分为三种,分别为共享锁,独占锁和更新锁.我们从程序的角度来看锁可以分为两种类型,悲观锁和乐观锁,Hibernate提供对这两种锁 的支持,我们来了解一下Hiber ...

  7. STM32F103 串口-IAP程序升级

    STM32F103 串口-IAP程序升级 通常情况下我们给STM32系列的单片机烧录程序文件的时候,使用SWD.J-link或者通过设置BOOT引脚后,使用串口进行程序下载,这样的方式直接一次性将程序 ...

  8. Dubbo源码分析:Invoker

    背景 调用对象!在调用过程可以使用Filter接口方法.Inovoker调用过程采用了装饰者设计模式.Filter最后一个ExcpetionFilter对象,这个对象之后就调用服务方法.服务对象是配置 ...

  9. Java中String、LocalDateTime、LocalDate、Date互转

    String 转LocalDate和LocalDateTime LocalDate startDate = LocalDate.parse("2019-12-05", DateTi ...

  10. [NPM + React] Prepare a Custom React Hook to be Published as an npm Package

    Before we publish our package, we want to make sure everything is set up correctly. We’ll cover vers ...