以前的代码
在开发过程中经常会使用if...else...
进行判断抛出异常、分支处理等操作。
这些if...else...
充斥在代码中严重影响了代码代码的美观,这时我们可以利用Java 8的Function接口来消灭if...else...
。
以前经常这么使用
if (...){
throw new RuntimeException("出现异常了");
}
if (...){
doSomethings();
} else {
doOtherThins();
}
接下来介绍几种优雅操作的方式
Function 函数式接口
需要使用到函数式接口来完成优雅操作,还不了解函数式接口的兄弟们要先学习下Java Function函数式接口编程。
概念:使用注解@FunctionalInterface
标识来检验,并且只包含一个抽象方法的接口是函数式接口。
函数式接口主要分为Supplier供给型函数、Consumer消费型函数、Runnable无参无返回型函数、Function有参有返回型函数。
Supplier供给型函数
get()方法 无参有返回值
Supplier的表现形式为不接受参数、只返回数据
@FunctionalInterface
public interface Supplier<T> {
T get();
}
Consumer消费型函数
accept()方法 有参无返回值
Consumer消费型函数和Supplier刚好相反。Consumer接收一个参数,没有返回值
@FunctionalInterface
public interface Consumer<T> {
void accept(T t);
......
}
Runnable无参无返回型函数
run()方法 无参无返回值
Runnable的表现形式为即没有参数也没有返回值
@FunctionalInterface
public interface Runnable {
public abstract void run();
}
Function有参有返回型函数。
apply()方法 有参有返回值
Function函数的表现形式为接收一个参数,并返回一个值。Supplier
、Consumer
和Runnable
可以看作Function的一种特殊表现形式
@FunctionalInterface
public interface Function<T, R> {
R apply(T t);
......
}
使用方式
处理抛出异常的if
1.定义函数
定义一个抛出异常的形式的函数式接口, 这个接口只有参数没有返回值是个消费型接口
@FunctionalInterface
public interface ThrowException {
/**
* @param msg 抛出异常信息
*/
void throwMessage(String msg);
}
2.判断方法,定义ThrowException内部实现
创建工具类TUtils并创建一个isThrow()方法,方法的返回值为刚才定义的函数式接口-ThrowException
。ThrowException
的接口实现逻辑为当参数b为true时抛出异常
public class TUtils {
public static ThrowException isThrow(Boolean b) {
return msg -> {
if (b) {
throw new RuntimeException(msg);
}
};
}
}
3.调用方式
调用工具类参数参数后,调用函数式接口的throwMessage
方法传入异常信息。当入参为false时不抛异常,程序正常执行
TUtils.isThrow(false).throwMessage("抛出异常啦~");
入参是true,将会抛出异常
处理if分支操作
1.定义函数
创建一个名为BranchHandle
的函数式接口,接口的参数为两个Runnable接口。这两个两个Runnable接口分别代表了为true或false时要进行的操作
@FunctionalInterface
public interface BranchHandle {
/**
*
* @param trueHandle 为true时,要进行的操作
* @param falseHandle false时的操作
*/
void trueOrFalseHandle(Runnable trueHandle, Runnable falseHandle);
}
2.判断方法,给予具体实现
创建一个名为isTrueOrFalse的方法,方法的返回值为刚才定义的函数式接口-BranchHandle
。
public static BranchHandle isTrueOrFalse(Boolean b) {
return (trueHandle, falseHandle) -> {
// b为true,执行trueHandler.run,否则执行falseHandler.run
if (b) {
trueHandle.run();
} else {
falseHandle.run();
}
};
}
3.调用方式
TUtils.isTrueOrFalse(true).trueOrFalseHandle(
() -> System.out.println("true,执行操作"),
() -> System.out.println("false,执行操作"));
入参是true,其他两个Runnable类型参数直接用lambda表达式即可
入参false
如果存在值执行消费操作,否则执行基于空的操作
例如,以字符串是否为空值来进行不同的业务操作。
1.定义函数
创建一个名为PresentOrElseHandler
的函数式接口,接口的参数一个为Consumer接口。一个为Runnable,分别代表值不为空时执行消费操作和值为空时执行的其他操作
/**
* 空值与非空值的分支处理
*/
@FunctionalInterface
public interface PresentOrElseHandler<T extends Object> {
/**
* @param action 值不为空执行的消费操作
* @param emptyAction 值为空执行的操作
*/
void presentOrElseHandle(Consumer<? super T> action, Runnable emptyAction);
}
2.编写判断方法
创建一个名为isBlankOrNoBlank
的方法,方法的返回值为刚才定义的函数式接口-PresentOrElseHandler
。
public static PresentOrElseHandler<?> isBlankOrNoBlank(String str) {
return (action, emptyAction) -> {
if (StrUtil.isNotBlank(str)) {
action.accept(str);
} else {
emptyAction.run();
}
};
}
3.调用方式
调用工具类参数参数后,调用函数式接口的presentOrElseHandle
方法传入一个Consumer
和Runnable
TUtils.isBlankOrNoBlank("hello world").presentOrElseHandle(
System.out::println,
() -> System.out.println("空字符串!"));
非空字符串入参
空字符串入参
以上3种方式,采用函数式接口简化代码,干掉代码里的if-else…、优雅抛出异常。3种方式仅是一种思路,也可以根据项目的需求功能点或者代码片段来拓展出别的处理方式~