JDK1.8新特性-Lambda表达式
程序员文章站
2022-06-10 16:02:46
Lambda 表达式 Lambda 表达式,也可称为闭包,它是推动 Java 8 发布的最重要新特性。 Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中)。 使用 Lambda 表达式可以使代码变的更加简洁紧凑。 语法 lambda 表达式的语法格式如下: (parameters ......
lambda 表达式
lambda 表达式,也可称为闭包,它是推动 java 8 发布的最重要新特性。
lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中)。
使用 lambda 表达式可以使代码变的更加简洁紧凑。
语法
lambda 表达式的语法格式如下:
(parameters) -> expression (parameters) ->{ statements; }
以下是lambda表达式的重要特征:
- 可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。
- 可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号。
- 可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
- 可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定明表达式返回了一个数值。
lambda 表达式实例
lambda 表达式的简单例子:
// 1. 不需要参数,返回值为 5 () -> 5 // 2. 接收一个参数(数字类型),返回其2倍的值 x -> 2 * x // 3. 接受2个参数(数字),并返回他们的差值 (x, y) -> x – y // 4. 接收2个int型整数,返回他们的和 (int x, int y) -> x + y // 5. 接受一个 string 对象,并在控制台打印,不返回任何值(看起来像是返回void) (string s) -> system.out.print(s)
在 java8tester.java 文件输入以下代码:
public class java8tester { public static void main(string args[]){ java8tester tester = new java8tester(); // 类型声明 mathoperation addition = (int a, int b) -> a + b; // 不用类型声明 mathoperation subtraction = (a, b) -> a - b; // 大括号中的返回语句 mathoperation multiplication = (int a, int b) -> { return a * b; }; // 没有大括号及返回语句 mathoperation division = (int a, int b) -> a / b; system.out.println("10 + 5 = " + tester.operate(10, 5, addition)); system.out.println("10 - 5 = " + tester.operate(10, 5, subtraction)); system.out.println("10 x 5 = " + tester.operate(10, 5, multiplication)); system.out.println("10 / 5 = " + tester.operate(10, 5, division)); // 不用括号 greetingservice greetservice1 = message -> system.out.println("hello " + message); // 用括号 greetingservice greetservice2 = (message) -> system.out.println("hello " + message); greetservice1.saymessage("nowcoder"); greetservice2.saymessage("google"); } interface mathoperation { int operation(int a, int b); } interface greetingservice { void saymessage(string message); } private int operate(int a, int b, mathoperation mathoperation){ return mathoperation.operation(a, b); } }
执行以上脚本,输出结果为:
$ javac java8tester.java $ java java8tester 10 + 5 = 15 10 - 5 = 5 10 x 5 = 50 10 / 5 = 2 hello nowcoder hello google
使用 lambda 表达式需要注意以下两点:
- lambda 表达式主要用来定义行内执行的方法类型接口,例如,一个简单方法接口。在上面例子中,我们使用各种类型的lambda表达式来定义mathoperation接口的方法。然后我们定义了saymessage的执行。
- lambda 表达式免去了使用匿名方法的麻烦,并且给予java简单但是强大的函数化的编程能力。
变量作用域
lambda 表达式只能引用标记了 final 的外层局部变量,这就是说不能在 lambda 内部修改定义在域外的局部变量,否则会编译错误。
在 java8tester.java 文件输入以下代码:
public class java8tester { final static string salutation = "hello! "; public static void main(string args[]){ greetingservice greetservice1 = message -> system.out.println(salutation + message); greetservice1.saymessage("nowcoder"); } interface greetingservice { void saymessage(string message); } }
执行以上脚本,输出结果为:
$ javac java8tester.java $ java java8tester hello! nowcoder
我们也可以直接在 lambda 表达式中访问外层的局部变量:
public class java8tester { public static void main(string args[]) { final int num = 1; converter<integer, string> s = (param) -> system.out.println(string.valueof(param + num)); s.convert(2); // 输出结果为 3 } public interface converter<t1, t2> { void convert(int i); } }
lambda 表达式的局部变量可以不用声明为 final,但是必须不可被后面的代码修改(即隐性的具有 final 的语义)
int num = 1; converter<integer, string> s = (param) -> system.out.println(string.valueof(param + num)); s.convert(2); num = 5; //报错信息:local variable num defined in an enclosing scope must be final or effectively final
在 lambda 表达式当中不允许声明一个与局部变量同名的参数或者局部变量。
string first = ""; comparator<string> comparator = (first, second) -> integer.compare(first.length(), second.length()); //编译会出错
函数既可以作为参数,也可以作为返回值,看一下下面这个例子
//这是常规的collections的排序的写法,需要对接口方法重写 public void test1(){ list<string> list =arrays.aslist("aaa","fsa","ser","eere"); collections.sort(list, new comparator<string>() { @override public int compare(string o1, string o2) { return o2.compareto(o1); } }); for (string string : list) { system.out.println(string); } } //这是带参数类型的lambda的写法 public void testlamda1(){ list<string> list =arrays.aslist("aaa","fsa","ser","eere"); collections.sort(list, (comparator<? super string>) (string a,string b)-> {return b.compareto(a); }); for (string string : list) { system.out.println(string); } } //这是不带参数的lambda的写法 public void testlamda2(){ list<string> list =arrays.aslist("aaa","fsa","ser","eere"); collections.sort(list, (a,b)->b.compareto(a)); for (string string : list) { system.out.println(string); } }
ending...