Java8 Easy Introduction — 1. Lambda

Java8 Easy Introduction — 1. Lambda

吹水

好吧,其实很早之前就打算开始写些博客,不过一直没有坚持下来。之前打得草稿也不知道到哪里去了,现在心血来潮,来献献丑,求指导指导。

好吧,这里我先废话两句:
1. 本人写博客纯属学习,我就把我的博客当成记记笔记,如果对你刚好有用,那真是感动;
2. 因为个人能力的原因,本人的博客内容可能会比较简单,老鸟勿喷,小鸟勿捧。实事求是,老鸟多指导,小鸟多点赞,谢谢!~~
3. 最后一点:本人写博客的目的除了记笔记以外,还是希望能给新手(好吧,我也是小鸟。。),一些入门的教程吧,所以尽量会写的简单、易懂,老鸟要是觉得没有深度,就Alt + <- 回退吧。

好吧,吹水完毕,开始我们的正式内容吧,Java8中的Lambda表达式。

前言

Java8在2014年由Oracle发布了正式版,其中比较主要的特性就是:Lambda,Stream,Optional以及一些时间处理上的类(LocalData,LocalTime等)。

到现在才来说Java8似乎有点晚了,因为再过不久Java9都要出来了。不过,童鞋们还是不用太着急,因为虽然技术出的很快,但是在生产环境(公司打码)中,更新并不会那么快,因为公司都要求 ,一般不会贸然使用最新的技术。

Java8 Easy Introduction这一个系列的文章,主要是针对有一些Java基础的童鞋,介绍Java8的一些基本特性,对java8有个基本的认识。

本章内容:
- 策略模式
- Lambda表达式和函数接口
- 函数描述符和使用Java自带函数接口
- Lambda简化写法

正式内容

策略模式

好吧,我们首先来讲一些什么是策略模式。

可能有的童鞋会产生疑问,为啥要先讲策略模式?如果心中产生这个疑问,那你就需要继续看下去了。

以下模拟一些数据,创建User类,其中包含:name,gender,age三个属性,以下省略相应的getter,setter方法(好吧,eclipse,在类中右键 -> source -> Genarate getter and setter可以自动生成,我又罗嗦了。)。

以下就是典型JavaBean对象:

public class User {
    public static enum Gender {
        MALE, FEMALE
    }
    private String name;
    private Gender gender;
    private Integer age;
    ....
}

模拟一些数据,包含10个User的List(这里我用了Junit测试来搞):

public class UserDemoTest {

    private List<User> userList = new ArrayList<>();

    @Before
    public void readyUserData() {
        userList.add(new User("Mike", Gender.MALE, 22));
        userList.add(new User("Bob", Gender.MALE, 22));
        userList.add(new User("Alice", Gender.FEMALE, 22));
        userList.add(new User("Terry", Gender.MALE, 22));
        userList.add(new User("Illies", Gender.FEMALE, 22));
        userList.add(new User("Jesiby", Gender.FEMALE, 22));
        userList.add(new User("Parker", Gender.MALE, 22));
        userList.add(new User("Avril", Gender.FEMALE, 22));
        userList.add(new User("Jerry", Gender.MALE, 22));
        userList.add(new User("Berry", Gender.FEMALE, 22));
    }
}

假设有一个场景:现在有一个List,里面存着10个User对象,老湿叫你筛选出其中所有性别为女的User对象。OK,最直接的解决方案:

@Test
public void screenUserList() {
    List<User> femaleUserList = new ArrayList<>();
    for (int i = 0; i < userList.size(); i++) {
        if (userList.get(i).getGender() == Gender.FEMALE) {
            femaleUserList.add(userList.get(i));
        }
    }
    System.out.println(femaleUserList.size());
}

但是在另外的一个地方,同样是userList,但是要筛选出年龄大于25的用户对象,这个时候,你会怎么办?

再写一次上面的代码,改一下条件?No,No,No,开发软件的原则之一就是:Don’t respeat youself,尽量避免重复代码。

其实仔细分析一下上面的过程,你会发现:代码的框架是不变的,只有判断的逻辑变了。我们可不可以将变化的部分抽取出来?当然可以。这就是策略模式:通过接口告诉方法要做什么操作。

定义一个接口UserClassifyOperation,其中只有一个方法:判断用户是否符合条件

public interface UserClassifyOperation {
    boolean judge(User user);
}

接着,我们创建一个方法,将遍历userList的基本骨架放在方法中,并且传入一个UserClassifyOperation对象(具体判断行为)来影响这个方法的执行结果:

private List<User> classifyUserList(List<User> userList, UserClassifyOperation classifyOperation) {
    List<User> resultUserList = new ArrayList<>();
    for (int i = 0; i < userList.size(); i++) {
        User user = userList.get(i);

        if (classifyOperation.judge(user)) {
            resultUserList.add(user);
        }
    }
    return resultUserList;
}

好吧,你已经把具体的判断行为和常规的遍历行为分离了,接着只需要在调用的时候指定具体的判断逻辑。一般会用匿名内部类去处理,如下:

@Test
public void screenUserList() {
    List<User> resultUserList = classifyUserList(userList, new UserClassifyOperation() {
        @Override
        public boolean judge(User user) {
            return user.getAge() > 25;
        }
    });
    System.out.println(resultUserList.size());
}

现在,你只需要控制方法参数就能控制这个函数的行为了,这就是策略模式。

小小见解:个人觉得传递匿名内部类也是实属无奈,因为Java不能直接传递函数,只能传递对象。就像这种只有一行的代码,就写一段匿名内部类,确实很“不干净”。不过,Java8之后有了Lambda就好多了。

Lambda表达式

什么是Lambda表达式?最直观的理解,就是一个函数

比如说,举个例子,创建Runnable对象,传统的匿名内部类:

Runnable r = new Runnable() {
    @Override
    public void run() {
        System.out.println("use anonymous inner class");
    }
};

而使用Lambda表达式,则是:

Runnable r = () -> {System.out.println("use lambda");}

是不是Lambda显得简洁许多?这也是为什么函数编程会流行起来的原因:更加抽象,更加简洁。

从上面的例子也可以看出一点:虽然可以使用Lambda表达式,但还是得依附在接口上。

Lambda表达式包含以下三个部分:
1. 参数部分,如:() , (Integer i);
2. ->,参数部分和方法之间的连接符;
3. { code },方法块,其中花括号不是必须的,如果指包含一条语句,并且语句的返回值就是方法的返回值,则可以省略花括号。接下来的例子会讲到。

接着讲回之前筛选用户的例子,如何结合策略模式和Lambda?改写之前筛选年龄的例子:

@Test
public void screenUserList() {
    List<User> resultUserList = classifyUserList(userList, (User user) -> user.getAge() > 25);
    System.out.println(resultUserList.size());
}

哇奥!5行变成2行,不得不说,Lambda对于方法代码少的情况,还是比匿名内部类的写法清晰很多。

在上面这种写法,实际上就等于创建了一个对象,传递给方法:

UserClassifyOperation u = (User user) -> user.getAge() > 25 ; 

这种写法实际上是和匿名内部类等效的。

那么,明明Lambda表达式更简洁,还要匿名内部类做啥?因为Lambda表达式只能生成函数接口的对象

那什么又是函数接口呢?其实最直观的理解,只有一个抽象方法的接口。比如之前的UserClassifyOperation接口:

public interface UserClassifyOperation {
    boolean judge(User user);
}

其实从Lambda的用法中,我们也可以看出:你只能指定一个参数列表,一个方法体。如果接口中包含多个方法,那么编译器是不知道你实现的是哪个方法的,所以Lambda只适用于函数接口。

注:在Java中,可以通过@FunctionalInterface来标记函数接口,这样编译器就会帮你做检查,如果超过一个抽象方法,就会抛出一个编译错误。同样,如果在看别人代码时,发现接口有这个注解,那就可以毫不犹豫地使用Lambda表达式了。

函数描述符和Java自带的函数接口

学习了如何使用Lambda表达式之后,我们现在来更进一步的了解一下:编译器是如何判断Lambda表达式编写正确?或者说,编译器是怎么知道Lambda表达式和接口是匹配的?

答案:通过函数描述符来进行判断。

那么什么是函数描述符?最直观的理解就是:参数 + 返回值,就是这个函数接口的函数描述符(函数接口只有一个方法)。

再回到我们刚才的UserClassifyOperation接口,以下对比一下其方法和对应的Lambda表达式:

boolean judge(User user);   // 方法

(User user) -> user.getAge() > 25 ; //lambda表达式 

函数方法接受User类型,并且返回boolean类型的方法,那么User + boolean就是这个函数接口的函数描述符。而我们编写Lambda表达式的时候,也同样会指定参数 + 返回值。这样编译器通过匹配Lambda表达式和函数接口的函数描述符是否一致,就可以判断你的Lambda表达式是否正确。

Lambda作用于多个接口

因为编译器判断Lambda是通过函数描述符来判断,即判断条件只与函数方法参数和返回值相关,和具体的接口,接口方法名无关。那么,假设我还有一个接口UserClassifyOperation2,其中方法也是返回boolean,并且接受User类型的话,同样的Lambda表达式也能作用到这个接口上。

正因为如此,Java已经为我们提供了一些更通用的函数接口,可以直接应用到我们的程序中,以下展示一个Predicate接口。可以到java.util.function中查看更多的函数接口:

@FunctionalInterface
public interface Predicate<T> {
    boolean test(T t);
}

简单来解释一下:Predicate接口,称为谓词。简单来说,就是判断一个值是否符合条件,其中的test方法就是判断方法。

因为Predicate可以接受泛型,所以我们完全可以用这个接口来替换UserClassifyOperation接口,修改筛选方法:

private List<User> classifyUserList(List<User> userList, Predicate<User> predicate) {
    List<User> resultUserList = new ArrayList<>();
    for (int i = 0; i < userList.size(); i++) {
        User user = userList.get(i);

        if (predicate.test(user)) {
            resultUserList.add(user);
        }
    }
    return resultUserList;
}

同样执行筛选年龄大于25岁的用户的方法:

@Test
public void screenUserList() {
    List<User> resultUserList = classifyUserList(userList, (User user) -> user.getAge() > 25);
    System.out.println(resultUserList.size());
}

可以看到,我们并没有修改Lambda表达式,它可以成功的和UserClassifyOperation以及Predicate接口匹配。再一次说明:同一个Lambda可以作用在多个函数接口上。

Lambda简化写法

先来讲一些Lambda的基本规则:

(User user) -> user.getAge() > 25 ; 

上面的式子指定了User参数,返回Boolean类型,下面的式子也是等效的:

(User user) -> {return user.getAge() > 25 ;}

之前讲过这个概念,不加花括号(就是代码块),就默认返回表达式的值;

省略参数类型

虽然将参数类型写出来,有利于编译器帮我们做检查。但是我相信,大多数情况下,你都知道自己在做什么。

所以,其实我们是可以省略参数类型的,如下:

// 单个参数可以省略()
Predicate<User> p = user -> user.getAge() > 25 ; 

// 多个参数必须使用()
Comparator<User> compare = (user1, user2) -> user1.getAge() - user2.getAge();

编译器知道我们要将Lambda表达式匹配到那个接口上,它会自己去推断参数的类型。这样,我们就可以省略参数类型。

使用现有方法引用

假设,你现在已经有一个现成的筛选方法,比如说:

public static boolean isAgeSuite(User user) {
    return user.getAge() > 25;
}

好吧,现在你可以更简化查询的函数了。

@Test
public void screenUserList() {
    List<User> resultUserList = classifyUserList(userList, UserDemoTest::isAgeSuite);
    System.out.println(resultUserList.size());
}

我们通过UserDemoTest::isAgeSuite引用了现有的方法,其实内部过程也不复杂。
1. Predicate接受到一个User类型的参数;
2. 将这个User参数,传递给指定的方法,此处为isAgeSuite;
3. 然后将isAgeSuite的返回值,作为Predicate中函数方法的返回值。

这就是两个最基本的Lambda表达式简化过程了,详细的可以去看《Java8实战》,里面讲得更详细。

小结

这一节,简单介绍了Lambda最基本的使用方式,如何与策略模式进行结合,了解编译器是如何检查Lambda表达式,以及如何简化Lambda表达式的写法。

其实也写了挺多东西的了,作为第一篇博客吧,写了有点久,不知道语言组织如何。反正如果有人还是看不懂就留言吧,有啥指导意见也可以留言提一提。

最后,非常感谢你看到这里!~~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值