@学习记录
开始学习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;
});
}
}