Java8 函数式编程

Java8 函数式编程

1.引言

Java 8的最大变化是引入了Lambda表达式,——一种紧凑的,传递行为的方式。它使开发者在写回调函数和事件处理程序时,不必纠缠于匿名内部类的冗繁和可读性,函数式编程让事件处理系统变得更加简。
Lambda表达式是一个匿名方法,将行为像数据一样传递。
Lambda表达式需要"函数式接口"的支持,
函数式接口:接口中只有一个抽象方法的接口称为函数式接口,可以使用注解@FunctionalInterace修饰。

2.Lambda表达式

2.1 Lambda表达式基本形式

// 1.不包含参数
//使用空括号()表示没有参数,返回类型为void
Runnable thread = () -> System.out.println("hello");

// 2.包含一个参数
// 如下表达式包含且只包含一个参数,可以省略参数的括号
ActionListener one = event -> System.out.println("hello");

// 3.表达式的主体是一段代码块
Runnable thread () -> {
  System.out.println("hello");
  System.out.println("hello1");
}

// 4.表达式包含多个参数
BinaryOperator<long> add = (x,y) -> x+y;

// 5.显示声明参数类型
BinaryOperator<long> add = (long x,long y) -> x+y;

2.2 java 8 内置四大函数式核心接口

Consumer<T> : 消费型接口

void accept(T t);

Supplier<T> : 供给型接口

T get();

Function<T, R> : 函数型接口

R apply(T t);

Predicate<T> : 断言型接口

boolean test(T t);

2.3 举个例子

  • 1.消费型接口
// 消费型接口
    @Test
    public void test1(){
        goHappy(1000.2,(m) -> System.out.println(m));
    }
    public void goHappy(Double money , Consumer<Double> consumer){
        consumer.accept(money);
    }
  • 2.供给型接口
//供给型接口
    @Test
    public void test2(){
        List<Integer> list = getNumList(10, () -> (int) (Math.random() * 100));
        list.stream().forEach(System.out::println);
    }

    public List<Integer> getNumList(int num , Supplier<Integer> supplier){
        List<Integer> list = new ArrayList<>();
        for(int i = 0;i< num; i++){
            Integer integer = supplier.get();
            list.add(integer);
        }
        return list;
    }
  • 3.函数型接口
// 函数型接口
    @Test
    public void test3(){
        String handler = Handler("\t\t 可口可乐 \t\t", str -> str.trim());
        System.out.println(handler);
    }
    public String Handler(String str , Function<String , String> function){
        return function.apply(str);
    }
  • 4.断言型接口
// 断言型接口
    @Test
    public void test4(){
        List<String> list = Arrays.asList("Hadoop" , "hive" , "HBase","Zookeeper" , "Spark");
        List<String> filter = filter(list, (str) -> str.length() > 5);
        filter.forEach(System.out::println);
    }

    public List<String> filter(List<String> list , Predicate<String> predicate){
        List<String> tmpList = new ArrayList<>();
        for (String s : list) {
            if(predicate.test(s)){
                tmpList.add(s);
            }
        }
        return tmpList;
    }

3.Stream

Stream是用函数式编程方式在集合类上进行复杂操作的工具。

Java 开发人员在处理集合时,一个通用的做法是在集合上进迭代,然后处理返回的每一个元素。
如,从ArrayList集合中统计字符"a"的出现次数,传统做法:

   Iterator<String> iterator = list.iterator();
        int count = 0;
        while (iterator.hasNext()){
            if(iterator.next().equals("a")){
                count++;
            }
        }

使用Stream:

long count = list.stream().filter(item -> item.equals("a")).count();

是不是很简洁!

3.1 Stream的创建

Stream共有四种创建方式:

 /**
         * 1.可以通过Collection 系列集合提供的stream()或parallelStream()
         */
         ArrayList<String> list = new ArrayList<>();
         Stream<String> stream1 = list.stream();

        /**
         * 2.通过Arrays中的静态方法stream()获取数组流
         */
         Stream<String> stream2 = Arrays.stream(new String[]{"a", "b", "c"});

        /**
         * 3.通过Stream类中的静态方法 of()
         */
         Stream<String> stream3 = Stream.of("a", "b", "c");

        /**
         * 4.创建无限流
         */
         Stream<Integer> stream4 = Stream.iterate(0, (x) -> x + 2);

3.2 Stream的筛选与切片操作

方法 描述
filter(Predicate p) 接收lambda,从中间排除某些元素
distinct() 筛选,通过流所生成的元素的hashCode()和equles()去除重复元素
limit(long maxSize) 截断流,使其元素不超过给定数量
skip(long n) 跳过元素,返回一个扔掉了前n个元素的流。若流中元素不足n个,则返回一个空值。与limit(n)互补
  • 1.filter
List<Employee> employees = Arrays.asList(
            new Employee("zhangsan", 19, 1000.0),
            new Employee("lisi", 20, 1002.0),
            new Employee("wangwu", 21, 1003.0),
            new Employee("xiaohong", 22, 1004.0),
            new Employee("xiaoming", 23, 1005.0)
    );

    @Test
    public void test1(){
        employees.stream()
                .filter(e -> e.getAge()>=21)
                .forEach(System.out::println);
    }
  • 2.limit
@Test
    public void test2(){
        employees.stream()
                .limit(2)
                .forEach(System.out::println);
    }
  • 3.skip
@Test
    public void test3(){
        employees.stream()
                .skip(2)
                .forEach(System.out::println);
    }

3.3 Stream的映射操作

方法 描述
map(Function f) 接收一个函数作为参数,该函数回被应用到每个元素上,并将其映射成一个新的元素
mapToDouble(ToDoubleFunction mapper) 接收一个函数作为参数,该函数回被应用到每个元素上,产生一个新的DoubleStream
mapToInt(ToIntFunction mapper) 接收一个函数作为参数,该函数回被应用到每个元素上,产生一个新的IntStream
mapToLong(ToLongFunction mapper) 接收一个函数作为参数,该函数回被应用到每个元素上,产生一个新的LongStream
flatMap(Function f) 接收一个函数作为参数,将流中的每一个值都转换成另一个流,然后把所有流都连接成一个流
  • 1.map
@Test
    public void test5(){
         List<String> list = Arrays.asList("aaa", "bbb", "ccc", "ddd");
         list.stream()
                 .map(str -> str.toUpperCase())
                 .forEach(System.out::println);
    }

3.4 Stream的排序操作

方法 描述
sorted() 产生一个新的流,其中按自然顺序排序
sorted(Comparator comp) 产生一个新的流,其中按比较器定义的规则排序
  • 1.sorted
@Test
    public void test6(){
        List<String> list = Arrays.asList("eee","aaa", "bbb", "ccc", "ddd");
        list.stream()
                .sorted()
                .forEach(System.out::println);
    }
  • 2.sorted(Compare comp)
 @Test
    public void test7(){
        employees.stream()
                .sorted((e1,e2) -> {
                    return e1.getName().compareTo(e2.getName());
                })
                .forEach(System.out::println);
    }

3.5 Stream的查找与匹配操作

方法 描述
allMatch(Predicate p) 检查是否匹配所有元素
anyMatch(Predicate p) 检查是否至少匹配其中一个元素
noneMatch(Predicate p) 检查是否没有匹配所有元素
findFirst() 返回第一个元素
findAny() 返回当前流中的任意元素
  • 1.allMatch
@Test
    public void test8(){
         boolean b = employees.stream()
                .allMatch(e -> e.getName().equals("lisi"));
        System.out.println(b);
    }
  • 2.anyMatch
@Test
    public void test9(){
        boolean b = employees.stream()
                .anyMatch(e -> e.getName().equals("lisi"));
        System.out.println(b);
    }
  • 3.noneMatch
@Test
    public void test10(){
        boolean b = employees.stream()
                .noneMatch(e -> e.getName().equals("lisi"));
        System.out.println(b);
    }
  • 4.findFirst
@Test
    public void test11(){
         Employee employee = employees.stream()
                .findFirst()
                .get();
        System.out.println(employee.toString());
    }

3.6 Stream的规约与收集操作

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

推荐阅读更多精彩内容