[代码简化] 1 Lambda表达式

1、lambda表达式
1.1 lambda

利用lambda表达式代替匿名参数类,将行为进行参数化传递到代码中,由代码动态调用。
举例,给定一个实体列表,根据实体的某个属性进行筛选。
<1> 这种情况下,最开始的做法就是:

    public static List<Sku> filterElectronicsSkus(List<Sku> skuList){
        List<Sku> result=new ArrayList<>();

        for (Sku sku:cartSkuList){
            if (SkuCategoryEnum.ELECTRONICS.equals(sku.getSkuCategory())){
                result.add(sku);
            }
        }
        return result;
    }

硬编码,进行指定条件筛选。

<2> 如果,再根据某个属性的其他值进行筛选,进一步做法:
除了原始列表,也会将条件作为参数之一

    public static List<Sku> filterSkusByCategory(List<Sku> skuList,SkuCategoryEnum category){
        List<Sku> result=new ArrayList<>();

        for (Sku sku:cartSkuList){
            if (category.equals(sku.getSkuCategory())){
                result.add(sku);
            }
        }

        return result;
    }

<3> 如果,不仅仅是根据category这一个属性,可能还会根据price等别的属性筛选:
先定义一个接口

public interface SkuPredicate {

    /**
     * 选择判断标签
     * @param sku
     */
    boolean test(Sku sku);
}

定义接口实现类:

public class SkuBooksCategoryPredicate implements SkuPredicate {
    @Override
    public boolean test(Sku sku) {
        return SkuCategoryEnum.BOOKS.equals(sku.getSkuCategory());
    }
}

public class SkuTotalPricePredicate implements SkuPredicate {
    @Override
    public boolean test(Sku sku) {
        return sku.getTotalPrice()>2000;
    }
}

之后,再出现别的筛选条件,只需要多定义一个接口实现类就可以了

 public void filterSkus() {
        List<Sku> cartSkuList = CartService.getCartSkuList();
        //过滤商品总价大于2000的商品
        List<Sku> res = CartService.filterSkus(cartSkuList, new SkuTotalPricePredicate());
        System.out.println(JSON.toJSONString(res, true));
    }

<4> 我们会发现,这样下去,接口实现类会越来越多,并且这些接口实现类往往复用的机会并不多,所以会想到使用匿名类代替。

public void filterSkus1() {
        List<Sku> cartSkuList = CartService.getCartSkuList();

        //过滤商品sku价大于2000的商品
        //todo 这里的接口实现类只会使用一次,所以可以在这里直接使用匿名内部类
        List<Sku> res = CartService.filterSkus(cartSkuList, new SkuPredicate() {
            @Override
            public boolean test(Sku sku) {
                return sku.getSkuPrice() > 3000;
            }
        });

        System.out.println(JSON.toJSONString(res, true));
    }

<5> 对于匿名内部类,我们可以使用lambda表达式来代替

public void filterSkus2() {
        List<Sku> cartSkuList = CartService.getCartSkuList();

        //过滤商品sku价大于2000的商品
        //todo 这里的接口实现类只会使用一次,所以可以在这里直接使用匿名内部类
        List<Sku> res = CartService.filterSkus(cartSkuList, sku -> sku.getSkuPrice() > 1000);

        System.out.println(JSON.toJSONString(res, true));
    }

从上面可以总结:函数编程演化历程:


Lambda表达式:
两种形式

  1. (parameters) -> expression
  2. (parameters) -> {statement;}

语法简写形式:
(1) 可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。 即()里面无需声明参数类型。()里面甚至可以什么都没有
(2) 可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号。 即()可以省略。
(3) 可选的大括号:如果主体包含了一个语句,就不需要使用大括号。 即{}可以省略,同时语句后面的分号也可以省略。
(4) 可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定明表达式返回了一个数值。

形式:
(1) 没有参数,小括号不能省略

() -> System.out.println("Hello World");

(2) 只有一个参数,小括号可省略

name -> System.out.println("Hello World"+name+"!");

(3) 没有参数,逻辑复杂

() ->
{
    System.out.print("Hello");
    System.out.println(" World");
};

(4)

// 创建一个函数
BinaryOperator<Long> functionAdd = (x, y) -> x + y;
// 应用函数,给函数传值,得到计算结果
Long result = functionAdd.apply(1L, 2L);

(5) 对参数类型的显示声明

// 创建一个函数
BinaryOperator<Long> function = (Long x, Long y) -> x + y;
// 应用函数,给函数传值,得到计算结果
Long result = function.apply(1L, 2L);
1.2 函数式接口:

允许将函数本身作为参数传入另一个函数。

  • 接口中有且只有一个抽象方法,但是可以有多个非抽象方法
  • Java8的函数式接口注解:@FunctionalInterface
    该注解可用于一个接口的定义上,一旦使用该注解来定义接口,编译器将会强制检查该接口是否确实有且仅有一个抽象方法,否则将会报错。需要注意的是,即使不使用该注解,只要满足函数式接口的定义,这仍然是一个函数式接口,使用起来都一样。

自定义函数示例:

@FunctionalInterface
public interface FileConsumer {

    void fileHandler(String fileContent);
}

public class FileService {

    public void fileHandle(String url, FileConsumer fileConsumer) throws IOException {

        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(new FileInputStream(url)));

        String line;
        StringBuilder stringBuilder = new StringBuilder();

        //循环读取文件内容
        while ((line = bufferedReader.readLine()) != null) {
            stringBuilder.append(line + "\n");
        }

        fileConsumer.fileHandler(stringBuilder.toString());
    }
}

public class FileServiceTest {
    @Test
    public void fileHandle() throws IOException {
        FileService fileService=new FileService();

        fileService.fileHandle("文件路径",fileContent -> System.out.println(fileContent));
    }
}

结果输出该指定文件路径下的文件内容。

常用函数接口:


在java.util.function包下。

1.3 方法引用

调用特定方法的Lambda表达式的一种快捷写法,可以让你重复使用现有的方法定义,并像Lambda表达式一样传递。



(1)指定静态方法的方法引用:
Integer::parseInt

 List<String>  strs= Lists.newArrayList("1","33","44","66");
 System.out.println(strs.stream().map(Integer::parseInt).max(Ordering.natural()).get());

(2)指向任意类型实例方法的方法引用:


(3)指向现有对象的实例方法的方法引用


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354

推荐阅读更多精彩内容