java8新特性之函数式编程范式

给定一个字符串元素列表,如下所示:

["1", "2", "bilibili", "of", "codesheep", "5", "at", "BILIBILI", "codesheep", "23", "CHEERS", "6"]

里面有数字型字符串,有字母型字符串;字符串里有大写,也有小写;字符串长度也有长有短

现在要写代码完成一个小功能

我想找出所有 长度>=5的字符串,并且忽略大小写、去除重复字符串,然后按字母排序,最后用“爱心❤”连接成一个字符串输出!

首先我写一个函数,判断输入字符串到底是字母还是数字

public static Boolean isNum( String str ) {
	for( int i=0; i<str.length(); i++ ) {
		if(!Character.isDigit(str.charAt(i))) {
			return false;
		}
	}
	return true;
}

接下来我一顿SAO操作:

// 先定义一个具备按字母排序功能的Set容器,Set本身即可去重
Set<String> stringSet = newTreeSet<String>(new Comparator<String>() {
		@Override
		public int compare(String o1, String o2) {
			return o1.compareTo(o2);  // 按字母顺序排列
		}
	}
);

// 以下for循环完成元素去重、大小写转换、长度判断等操作
for( int i=0; i<list.size(); i++ ) {
	String s = list.get(i);
	if( !isNum(s) && s.length()>=5) {
		String sLower = s.toLowerCase(); 
		// 统一转小写
		stringSet.add( sLower );
	}
}

// 以下for循环完成连词成句
StringBuilder result = new StringBuilder();
for( String s : stringSet ) {
    result.append( s );
    result.append("❤"); // 用“爱心”连接符连接
}
String finalResult = result.substring(0,result.length()-1).toString();  // 去掉最后一个多余连接符
System.out.println( finalResult );

最后输出结果为:

bilibili❤cheers❤codesheep

针对上面的作业,用Java 8的 Stream流式操作,仅需一行代码就可以搞定,for循环啥的统统灰飞烟灭。

String result = list.stream()// 首先将列表转化为Stream流
.filter( i -> !isNum(i) )// 首先筛选出字母型字符串
.filter( i -> i.length() >= 5)// 其次筛选出长度>=5的字符串
.map( i -> i.toLowerCase() )// 字符串统一转小写
.distinct()                 // 去重操作来一下
.sorted( Comparator.naturalOrder() ) // 字符串排序来一下
.collect( Collectors.joining("❤") ); // 连词成句来一下,完美!

System.out.println(result);

上面其实已经通过举栗的方式阐述了Java 8函数式编程范式:Stream流 的优雅和强大,尤其在处理集合时,几本一步到位,嘎嘣脆。

当然Stream也仅仅只是Java 8函数式编程接口的一个而已,除了Stream接口,还有其他非常强大的函数式编程接口,比如:

  • Consumer接口
  • Optional接口
  • Function接口

一、Consumer接口

顾名思义,它是“消费者的含义”,接受参数而不返回值,举个最最常见的栗子:

平时我们打印字符串,本质也是接受一个参数并打印出来,我们一般想都不想,会这样写:

System.out.println("hello world");     // 打印 hello world
System.out.println("hello codesheep"); // 打印 hello codesheep
System.out.println("bilibili cheers"); // 打印 bilibili cheers

一旦你用了 Consumer之后,总感觉更加优雅一些

Consumer c = System.out::println;

c.accept("hello world");      // 打印 hello world
c.accept("hello codesheep");  // 打印 hello codesheep
c.accept("bilibili cheers");  // 打印 bilibili cheers

而且 Consumer还可以用联用,达到多重处理的效果,比如:

c.andThen(c).andThen(c).accept("hello world");
// 会连续打印 3次:hello world

当然本例只是打印字符串,比较简单,若业务更加复杂, Consumer复用带来的便利性还是不小的。

二、Function接口

Function接口代表的含义是“函数”,其实和上面的 Consumer有点像,不过 Function既有输入,也有输出,使用更加灵活,举例:

比如我想对一个整数先乘以 2,再计算平方值

Function<Integer,Integer> f1 = i -> i+i;  // 乘以2功能
Function<Integer,Integer> f2 = i -> i*i;  // 平方功能
Consumer c = System.out::println;         // 打印功能

c.accept( f1.andThen(f2).apply(2) );      // 三种功能组合:打印结果 16

别的不说,这个炫技操作还是可以的!

三、Optional接口

Optional本质是个容器,你可以将你的变量交由它进行封装,这样我们就不用显式对原变量进行 null值检测,防止出现各种空指针异常。举例:

我们想写一个获取学生某个课程考试分数的函数:getScore()

public Integer getScore( Student student ) {
	if( student != null) {      // 第一层 null判空
		Subject subject = student.getSubject();
		if( subject != null) {  // 第二层 null判空
			return subject.score;
		}
	}
	return null;
}

这样写倒不是不可以,但我们作为一个“严谨且良心的”后端工程师,这么多嵌套的 if 判空多少有点扎眼!

为此我们必须引入 Optional:

public Integer getScore( Student student ) {
	return Optional
	.ofNullable(student)
	.map( Student::getSubject )
	.map( Subject::getScore )
	.orElse(null);
}

漂亮!嵌套的if/else判空灰飞烟灭!

发布了156 篇原创文章 · 获赞 8 · 访问量 2万+

猜你喜欢

转载自blog.csdn.net/weixin_42590334/article/details/103957552