Java SE 学习记录——进阶版12

@学习记录

开始学习Java

遵从同学的指导,从Java se开始学习

黑马的JavaSE零基础入门【函数式接口】

第一章 函数式接口

1.1 概念
1.2 格式
1.3 @FunctionInterface注释
1.4 自定义函数式接口
23-02 函数式接口的概念&函数式接口的定义
package day_12FunctionInterface.demo01FunctionInterface;

/*
函数式接口:有且只有一个抽象方法的接口
可以包含其他的方法(默认,静态,私有)

@FunctionalInterface 注解
作用:可以检查接口是否是一个函数式接口
    是:编译成功
    否:编译失败(接口中没有抽象方法;抽象方法的个数多于一个)
 */

@FunctionalInterface
public interface MyFunctionInterface {
    // 定义一个抽象方法
    public abstract void method();
}

package day_12FunctionInterface.demo01FunctionInterface;

public class MyFunctionInterfaceImpl implements MyFunctionInterface{
    @Override
    public void method() {

    }
}


23-03 函数式接口的使用
package day_12FunctionInterface.demo01FunctionInterface;

/*
函数式接口的使用:一般可以作为方法的参数和返回值类型
 */
public class Demo {
    // 定义一个方法,参数使用函数式接口MyFunctionInterface
    public static void show(MyFunctionInterface mfi) {
        mfi.method();
    }

    public static void main(String[] args) {
        // 调用show方法
        show(new MyFunctionInterfaceImpl());

        // 传递接口的匿名内部类
        show(new MyFunctionInterface() {
            @Override
            public void method() {
                System.out.println("使用匿名内部类重写接口的抽象方法");
            }
        });

        // 传递Lambda表达式
        show(()-> System.out.println("传递Lambda表达式"));
    }
}


第二章 函数式编程

2.1 Lambda的延迟执行

性能浪费的日志案例
体验Lambda的更优写法
证明Lambda的延迟

23-04 性能浪费的日志案例
package day_12FunctionInterface.demo02Lambda;

/*
日志案例

发现以下代码存在性能浪费
先拼接好字符串再调用showLog方法,如果日志的等级不是1,则不会输出拼接后的字符串,存在性能的浪费
 */
public class Demo01Logger {
    // 定义一个根据日志的级别写是日志信息的方法
    public static void showLog(int level, String message) {
        // 对日志的等级进行判断,如果是1,输出日志信息
        if (level == 1) {
            System.out.println(message);
        }
    }

    public static void main(String[] args) {
        // 定义三个日志信息
        String msg1 = "法外";
        String msg2 = "狂徒";
        String msg3 = "张三";

        // 调用showLog方法,传递日志级别和日志信息
        showLog(2,msg1+msg2+msg3);
    }
}


23-05 使用Lambda优化日志案例
package day_12FunctionInterface.demo02Lambda;

/*
使用Lambda优化日志案例
Lambda的特点:延迟加载
Lambda的使用前提:必须存在函数式接口
 */
public class Demo02Lambda {
    // 定义一个显示日志的方法,方法的参数传递日志的等级和MessageBuilder接口
    public static void showLog(int level, MessageBuilder mb) {
        // 对日志的等级进行判断,如果是1,输出日志信息
        if (level == 1) {
            System.out.println(mb.builderMessage());
        }
    }

    public static void main(String[] args) {
        // 定义三个日志信息
        String msg1 = "法外";
        String msg2 = "狂徒";
        String msg3 = "张三";

        // 调用showLog方法,传递日志级别和日志信息
        showLog(1, ()-> msg1+msg2+msg3);
    }
}

package day_12FunctionInterface.demo02Lambda;

@FunctionalInterface
public interface MessageBuilder {
    // 定义一个拼接消息的抽象方法,返回被拼接的消息
    public abstract String builderMessage();
}


2.2 使用Lambda作为参数和返回值
23-06 函数式接口作为方法的参数案例
package day_12FunctionInterface.demo03LambdaTest;

/*
java.lang.Runnable接口就是一个函数式接口
假设有一个startThread方法使用该接口作为参数,可以用Lambda进行传参
 */
public class Demo01Runnable {
    // 定义一个startThread方法使用Runnable接口作为参数
    public static void startThread(Runnable runnable) {
        // 开启多线程
        new Thread(runnable).start();
    }

    public static void main(String[] args) {
        startThread(()-> System.out.println(Thread.currentThread().getName() + "启动") );
    }
}


23-07 函数式接口作为方法的返回值类型案例
package day_12FunctionInterface.demo03LambdaTest;

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 (o1,o2)->o2.length()-o1.length();
    }

    public static void main(String[] args) {
        // 创建一个字符串数组
        String[] arr = {"cj","hsg","a"};
        // 输出排序前的数组
        System.out.println(Arrays.toString(arr));
        // 调用Arrays中的sort方法对字符串数组排序
        Arrays.sort(arr,getComparator());
        // 输出排序后的数组
        System.out.println(Arrays.toString(arr));
    }
}


第三章 常用函数式接口

3.1 Supplier接口
23-08 常用函数式接口-Supplier接口
package day_12FunctionInterface.demo04Supplier;

import java.util.function.Supplier;

/*
常用的函数式接口
java.util.function.Supplier<T> 接口仅包含一个无参的方法:T get(),用来获取一个泛型参数指定类型的对象数据

Supplier<T> 接口被称之为生产型接口,指定接口的泛型是什么类型,那么接口中的get方法就会生产什么类型的数据
 */
public class Demo01Supplier {
    // 定义一个方法,方法的参数传递Supplier<T>接口,泛型指定String,get方法就会返回String
    public static String getString(Supplier<String> supplier){
        return supplier.get();
    }

    public static void main(String[] args) {
//        String s = getString(() -> {
//            // 生产一个字符串并返回
//            return "胡歌";
//        });

        // 优化
        String s = getString(() -> "胡歌");
        System.out.println(s);
    }
}


3.2 练习:求数组元素最大值

题目
解答

23-09 常用函数式接口-Supplier接口练习-求数组元素最大值
package day_12FunctionInterface.demo04Supplier;

import java.util.function.Supplier;

/*
练习:求数组元素的最大值
    使用Supplier接口作为方法参数类型,通过Lambda表达式求出int数组中的最大值
    提示:接口的泛型请使用java.lang.Integer类
 */
public class Demo02Test {
    // 定义一个方法,用于获取int类型数组中元素的最大值,方法的参数传递Supplier接口,泛型使用Integer
    public static int getMax(Supplier<Integer> supplier) {
        return supplier.get();
    }

    public static void main(String[] args) {
        int[] arr = {4,26,56,45,7,2};
        int maxNum = getMax(() -> {
            // 获取数组的最大值并返回
            int max = arr[0];
            for (int i = 0; i < arr.length; i++) {
                if (arr[i] > max) {
                    max = arr[i];
                }
            }
            return max;
        });
        System.out.println("最大值是" + maxNum);
    }
}


3.3 Consumer接口

抽象方法accept
默认方法andThen

23-10 常用函数式接口-Consumer接口
package day_12FunctionInterface.demo05Consumer;

import java.util.function.Consumer;

/*
java.util.function.Consumer<T>接口则正好与Supplier相反
    它不是生产一个数据,而是消费一个数据,其数据类型由泛型决定
Consumer接口中包含抽象方法void accept(T t),意为消费一个指定泛型的数据

Consumer接口是一个消费型接口,泛型指定什么类型,就可以使用accept方法消费什么类型的数据
具体消费(使用),需要自定义(输出,计算...)
 */
public class Demo01Consumer {
    public static void getConsumer(Consumer<String> consumer, String name) {
        consumer.accept(name);
    }

    public static void main(String[] args) {
        getConsumer(name->{
            // 消费方式:直接输出字符
            System.out.println(name);

            // 消费方式:把字符串进行反转输出
            String s = new StringBuffer(name).reverse().toString();
            System.out.println(s);
        },"张三");
    }
}


23-11 常用函数式接口-Consumer接口的默认方法andThen
package day_12FunctionInterface.demo05Consumer;

import java.util.function.Consumer;

/*
Consumer接口的默认方法andThen
作用:需要两个Consumer接口,可以把两个Consumer接口组合到一起,再对数据进行消费

例如:
    Consumer<String> con1           等价于         con1.andThen(con2).accept(s)谁写前面谁先消费
    Consumer<String> con2
    String s = "Hello"
    con1.accept(s)
    con2.accept(s)


 */
public class Demo02AndThen {
    public static void method(String s, Consumer<String> consumer1, Consumer<String> consumer2) {
//        consumer1.accept(s);
//        consumer2.accept(s);

        // 使用andThen方法把两个接口连接到一起再消费数据
        consumer1.andThen(consumer2).accept(s);
    }

    public static void main(String[] args) {
        method("xnCzo", (s)->{
            // 把字符串转换为大写输出
            System.out.println(s.toUpperCase());
        },(s)->{
            // 把字符串转换为小写输出
            System.out.println(s.toLowerCase());
        });
    }
}


3.4 练习:格式化打印信息

题目
解答

23-12 常用函数式接口-Consumer接口练习-字符串拼接输出
package day_12FunctionInterface.demo05Consumer;

import java.util.function.Consumer;

/*
练习:
    字符串数组中存有多条信息,请按照格式“姓名:xx。性别:xx。"的格式将信息打印出来
    要求将打印姓名的动作作为第一个Consumer接口的Lambda实例
    要求将打印性别的动作作为第二个Consumer接口的Lambda实例
    将两个Consumer接口按照顺序拼接到一起
 */
public class Demo03Test {
    public static void printMessage(String[] arr, Consumer<String> con1, Consumer<String> con2){
        for (String s : arr) {
            con1.andThen(con2).accept(s);
        }
    }

    public static void main(String[] args) {
        // 定义一个字符串类型的数组
        String[] arr = {"法外,20","狂徒,39","张三,40"};
        printMessage(arr,(s)->{
            String[] strings = s.split(",");
            System.out.print("姓名:" + strings[0] + "。");
        },(s)->{
            String[] strings = s.split(",");
            System.out.println("性别:" + strings[1] + "。");
        });
    }
}


3.5 Predicate接口

抽象方法 test
默认方法and
默认方法or
默认方法negate

23-13 常用函数式接口-Predicate接口
package day_12FunctionInterface.demo06Predicate;

import java.util.function.Predicate;

/*
java.util.function.Predicate<T> 接口
作用:对某种数据类型的数据进行判断,结果返回一个boolean值

Predicate接口中包含一个抽象方法
    boolean test(T t) 用来对指定数据类型数据进行判断的方法
        结果:
            符合条件 true
            不合符条件 false
 */
public class Demo01Predicate {
    // 定义一个方法,参数传递一个String类型的字符串,传递一个Predicate接口,泛型使用String,使用方法test对字符串进行判断,并把判断的结果进行返回
    public static boolean judge(String s, Predicate<String> p) {
        return p.test(s);
    }

    public static void main(String[] args) {
        boolean b = judge("cnxo", (s) -> s.length() == 3);
        System.out.println(b);
    }
}


23-14 常用函数式接口-Predicate接口-默认方法and
package day_12FunctionInterface.demo06Predicate;

import java.util.function.Predicate;

/*
逻辑表达式:可以链接多个判断的条件
&&:与运算符,有false则false
||:或运算符,有true则true
!:非(取反)运算符,非真即假,非假即真

需求:判断一个字符串,有两个判断条件
    1.判断字符串的长度是否大于5
    2.判断字符串中是否含有a
两个条件必须同时满足,可以使用&&运算符连接两个条件

Predicate接口中有一个方法and,表示并且关系,也可以链接两个判断条件
 */
public class Demo02Predicate_and {
    // 定义一个方法,方法的参数传递一个字符串,传递两个Predicate接口
    public static boolean checkString(String s, Predicate<String> p1, Predicate<String> p2) {
//        return p1.test(s) && p2.test(s);
        return p1.and(p2).test(s);
    }

    public static void main(String[] args) {
        boolean b = checkString("noidf", (s) -> {
            return s.length() > 5;
        }, (s) -> {
            return s.contains("a");
        });
        System.out.println(b);
    }
}


23-15 常用函数式接口-Predicate接口-默认方法or&negate
package day_12FunctionInterface.demo06Predicate;

import java.util.function.Predicate;

/*
需求:判断一个字符串,有两个判断条件
    1.判断字符串的长度是否大于5
    2.判断字符串中是否含有a
满足一个条件即可,可以使用||运算符连接两个条件

 */
public class Demo03Predicate_or {
    // 定义一个方法,方法的参数传递一个字符串,传递两个Predicate接口
    public static boolean checkString(String s, Predicate<String> p1, Predicate<String> p2) {
        return p1.or(p2).test(s);
    }

    public static void main(String[] args) {
        boolean b = checkString("hnoidf", (s) -> {
            return s.length() > 5;
        }, (s) -> {
            return s.contains("a");
        });
        System.out.println(b);
    }
}

package day_12FunctionInterface.demo06Predicate;

import java.util.function.Predicate;

/*
需求:判断字符串的长度是否大于5
    1.大于,返回false
    2.小于,返回true

 */
public class Demo04Predicate_negate {
    // 定义一个方法,方法的参数传递一个字符串,传递两个Predicate接口
    public static boolean checkString(String s, Predicate<String> p) {
        return p.negate().test(s);
    }

    public static void main(String[] args) {
        boolean b = checkString("hnoidf", (s) -> {
            return s.length() > 5;
        });
        System.out.println(b);
    }

}


3.6 练习:重合信息筛选

题目
解答

23-16 常用函数式接口-Predicate接口练习-重合信息筛选
package day_12FunctionInterface.demo06Predicate;

import java.util.ArrayList;
import java.util.function.Predicate;

/*
练习:集合信息的筛选
数组中有多条“姓名+性别”的信息如下:
String[] arr = {"迪士尼发泡,男", "都不能,女", "建筑室内哈佛请问,女", "找刺激,男"};
请通过Predicate接口的拼装将符合要求的字符串筛选到集合ArrayList中
需要同时满足两个条件:
    1.必须为女生
    2.姓名为4个字
 */
public class Demo05Test {
    public static ArrayList<String> filter(String[] arr, Predicate<String> p1, Predicate<String> p2) {
        // 存储筛选后的信息
        ArrayList<String> list = new ArrayList<>();
        for (String s : arr) {
            boolean b = p1.and(p2).test(s);
            if (b == true) {
                list.add(s);
            }
        }
        return list;
    }

    public static void main(String[] args) {
        String[] arr = {"迪士尼发泡,男", "都不能就,女", "建筑室内哈佛请问,女", "找刺激,男"};
        ArrayList<String> list = filter(arr, (s) -> {
            return s.contains("女");
        }, (s) -> {
            return s.length() ==  6;
        });
        for (String s : list) {
            System.out.println(s);
        }
    }
}


3.7 Function接口

抽象方法apply
默认方法andThen

23-17 常用函数式接口-Function接口
package day_12FunctionInterface.demo07Function;

import java.util.function.Function;

/*
java.util.function.Function<T,R> 接口用来根据一个类型的数据得到另一个类型的数据
    前者称为前置条件,后者称为后置条件
Function接口中主要的抽象方法为 R apply(T t),根据类型T的参数获取类型R的结果
    使用的场景例如:将String类型转换为Integer类型
 */
public class Demo01Function {
    public static void change(String s, Function<String,Integer> function) {
        Integer in = function.apply(s);
//        int in = function.apply(s); // 自动拆箱
        System.out.println(in);
    }

    public static void main(String[] args) {
        change("57234",(s)->{
            return Integer.parseInt(s);
        });
    }
}


23-18 常用函数式接口-Function接口-默认方法andThen
package day_12FunctionInterface.demo07Function;

import java.util.function.Function;

/*
Function接口中的默认方法andThen,用来进行组合操作

需求:
    把String类型的“123”转换为Integer类型,把转换后的结果加10
    把转换后的Integer类型的数据转换为String类型
 */
public class Demo02Function_andThen {
    public static void change(String str, Function<String,Integer> fun1, Function<Integer,String> fun2){
        String s = fun1.andThen(fun2).apply(str);
        System.out.println(s);
    }

    public static void main(String[] args) {
        change("1234",(s)->{
            return Integer.parseInt(s) + 10;
        },(in)->{
            return in + ""; // 将Integer类型的数据转换为String类型
        });
    }
}


3.8 练习:自定义函数模型拼接

题目
解答

23-19 常用函数式接口-Function接口练习-自定义函数模型拼接
package day_12FunctionInterface.demo07Function;

import java.util.function.Function;

/*
练习:自定义函数模型拼接
请使用Function进行函数模型的拼接,按照顺序需要执行多个函数操作为:
    String str = "张三,30";
    1.将字符串截取数字年龄部分,得到字符串
    2.将上一步的字符串转换为int类型的数据
    3.将上一步int数字累加100,得到结果int数字
 */
public class Demo03Test {
    public static void change(String str, Function<String,String> fun1, Function<String,Integer> fun2, Function<Integer,Integer> fun3 ){
        int num = fun1.andThen(fun2.andThen(fun3)).apply(str);
        System.out.println(num);
    }

    public static void main(String[] args) {
        change("张三,30",(str)->{
            return str.split(",")[1];
        },(str)->{
            return Integer.parseInt(str);
        },(in)->{
            return in + 100;
        });
    }
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值