您的位置:首页 > 编程语言 > Java开发

初探Java8lambda表达式

2015-12-03 16:43 549 查看
要说现在什么火, 估计函数式编程算一个吧,看看人家javascript写起来多爽,java呢?一切皆对象。好像离着函数式编程挺远的,不过在java8中我们终于迎来了类似函数式编程-Java风格的lambda表达式,在用lambda重构你的代码后,啥感觉? 倍爽!

这篇博客我们就来小探一下java8的lambda表达式,领略一下lambda表达式的风骚!

lambda表达式语法

lambda的语句可以用以下伪代码表示,很简单,不过还有很多更简单的表达方式,

(Type ...argument) -> {code}


还有一下变种方式,

(Type ...argument) -> x * y // 执行x * y , 注意有返回值的情况下,这里会自动推导返回


初探lambda表达式

只看语法没什么卵用,最好的学习方法还是在实践中去掌握它。先来几个简单的尝尝鲜吧:用lambda表达式去遍历一个List。在没有lambda之前,我们是怎么遍历一个List的呢?相信大家很熟悉,

List<String> list = Arrays.asList("aaa", "bbb", "ccc");
for(String item : list) {
System.out.println(item);
}


不多说, 再来看看lambda怎么做,

List<String> list = Arrays.asList("aaa", "bbb", "ccc");
list.forEach((item) -> System.out.println(item));


一行代码搞定!关键代码就是
(item) -> System.out.println(item)
,这样的语法在上面我们已经介绍了,因为只有一句表达式,所以
{}
可以省略。

等等,上面的方式还不是最简单的,

List<String> list = Arrays.asList("aaa", "bbb", "ccc");
list.forEach(System.out::println);


哎哟,这什么玩意! 嘿嘿,System.out::println等同于
(x) -> System.out.println(x)
。这种语法我们在下面会详细讲解。

和上面遍历相似,我们再来看看线程现在该怎么写,

new Thread(() -> {
// do something
System.out.println("hello thread");
}).start();


恩,简单不? 而且有了上面的基础,也很好理解了。

排序!排序!

在lambda中排序也简单了许多,看下面代码,

String[] array = {"bbb", "ddd", "aaa"};
Arrays.sort(array, (String a, String b) -> a.compareTo(b));


stream

在java8中,Collection多了一个stream方法,stream方法天生就是配合lambda来的,stream提供了好多好用的方法,例如filter用来过滤,下面的代码,我们只打印以a开头的字符串。

List<String> list = Arrays.asList("abc", "abb", "ddd");
list.stream().filter((String item) -> item.startsWith("a"))
.forEach(System.out::println);


limit,类似sql中的limit,用来限制遍历个数,例如下面代码,我们将遍历List中的前5条数据,

List<String> list = Arrays.asList("abc", "abb", "ddd", "111", "ddd", "yyy", "999");
list.stream().limit(5).forEach(System.out::println);


关于Stream更多好玩的方法,可以去api中查看说明,这里就不一一讲解了。

自定义函数式接口

在上面的代码中,我们都是去使用了系统提供了函数式接口,那我们自己能不能自定义呢?答案是可以的,要自定义函数式接口必须符合以下两个条件。

该接口中有且仅有一个方法。

在接口上使用注解@FunctionalInterface (非必须,但良好的编程规范是加上)

遵循上面的规范,我们来自定义一个函数式接口吧,

@FunctionalInterface
interface Func {
String builder(String firstName, String lastName);
}


哎(二声),一个函数接口搞定了,如何用呢?其实你已经会用了。

Func func = (String first, String last) -> {
return first + "." + last;
};
System.out.println(func.builder("f", "l"));


恩,太熟悉了,不过估计有朋友想到了更简洁的代码,

Func func = (String first, String last) -> first + "." + last;
System.out.println(func.builder("f", "l"));


俩冒号是个啥

上面我们再用System.out::println的时候说了,System.out::println等同于
(x) -> System.out.println(x)
。现在就让我们来了解一下这个
::


::
其实是一种依赖lambda自动推导的语法糖(纳尼? 其实lambda表达式本身在java中就是一种语法糖),概念性的东西我也不会说,下面就让我们用3个例子来学习

一下这个
::
吧。

class Util {
public static String getSomething(String a, String b) {
return a + ";" + b;
}
}

@FunctionalInterface
interface Func {
public String get(String a, String b);
}

Func f = Util::getSomething;
System.out.println(f.get("aaa", "bbb"));


主要的代码就是
Func f = Util::getSomething
了,其实他就等同于
Func f = (String a, String b) -> Util.getSomething(a, b)
。其实就是用Util.getSomething()去实现了Func的get方法,不过这里看起来是不是有种Util实现了Func接口的感觉,嘿嘿。

上面的getSomething是一个静态方法,如果是实例方法呢?

class Util {
public String doGet(String a, String b) {
return a + ";" + b;
}
}
Util util = new Util();
Func f2 = util::doGet;
System.out.println(f2.get("ccc", "ddd"));


::
和构造方法呢?

class Student {
private String name;

public Student(String name) {
this.name = name;
System.out.println(this.name);
}
}

@FunctionalInterface
interface StudentInterface {
Student build(String name);
}

StudentInterface si = Student::new;
Student stu = si.build("loader");


像不像静态工厂模式?恩?
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: