浅析匿名函数、lambda表达式、闭包(closure)区别与作用




浅析匿名函数、lambda表达式、闭包(closure)区别与作用

浅析匿名函数、lambda表达式、闭包(closure)区别与作用

所有的主流编程语言都对函数式编程有支持,比如c++11、python和java中有lambda表达式、lua和JavaScript中有闭包(closure)、ObjectC中有块(blocks,^)。那么这几个概念有什么共性和区别呢,只用普通函数不行吗,为什么要创造这些炫酷的概念呢?

一.匿名函数、lambda、closure区别

从功能性上说lambda和closure(或是OC中的blocks)是一个东西,只是不同语言的不同称呼罢了,它们都是匿名函数。若匿名函数捕获了一个外部变量,那么它就是一个closure(闭包)。

二.匿名函数、lambda、closure作用

简单说,引入它们的作用有2个:

  • 简洁
  • 捕获变量

首先说 简洁 ,匿名函数可以在其他函数内部声明与定义,不用另外写个命名函数,举个栗子,显示c++vector中每个值,若不使用匿名函数,代码是这样的

 1 #include <iostream>
 2 #include <vector>
 3 #include <algorithm>
 4 using namespace std;
 5 
 6 void show(int & value) {
 7     std::cout << "value " << value << std::endl;
 8 }
 9 
10 int main() {
11     vector<int> * v = new vector<int>(3, 1);
12 
13     for_each(v->begin(), v->end(), show);
14 
15     return 0;
16 }

若使用匿名函数是这样的:

 1 #include <iostream>
 2 #include <vector>
 3 #include <functional>
 4 #include <algorithm>
 5 using namespace std;
 6 
 7 int main() {
 8     vector<int> * v = new vector<int>(3, 1);
 9 
10     for_each(v->begin(), v->end(), [](int & v){
11         std::cout << "value " << v << std::endl;
12     });
13 
14     return 0;
15 }

什么是 捕获变量 呢?就是让匿名函数可以使用匿名函数外定义的变量,但是匿名函数内的函数外变量是外部变量的一个clone(C++11可以捕获引用,不是clone。OC中有__block,也可以直接改变外部变量的值),在匿名函数内修改(诺能)外部变量不会影响到外部变量。可以说closure就是函数+捕获的变量。以lua为例:

 1 local intValue = 10;
 2 
 3 local func = function (p)
 4     intValue = intValue + p;
 5     return intValue;
 6 end
 7 
 8 print(func(3));
 9 print(func(3));
10 print(intValue)

打印的结果为:13,16,10。可见匿名函数的外部变量被捕获到了func中。

三.匿名函数、lambda、closure在各个语言中的使用方式:

1.ObjectC:

在ObjectC中,匿名函数被称为blocks(块), 即可以改变捕获的原值、又可以捕获克隆、但不能改变克隆值的值 。捕获并改变外部值,需要用__block,否则复制语句会报错,使用代码如下:

 1 #import <Foundation/Foundation.h>
 2 
 3 int main(int argc, const char * argv[]) {
 4     @autoreleasepool {
 5         __block int foo = 10;
 6         
 7         int (^blockFunc)(int p) = ^(int p) {
 8             foo += p;
 9             return foo;
10         };
11         
12         NSLog(@"%d", blockFunc(4));
13         
14         NSLog(@"%d", foo);
15     }
16     return 0;
17 }  
2.C++:

C++中匿名函数被称为lambda, 即可以改变捕获的原值、又可以捕获克隆、又可以改变克隆值的值 ,语法形式可以简单归纳如下:

[capture](params)ops->ret{body;}

capture是捕获列表,params是参数表,opt是可选选项,ret是返回值类型,body是函数体。具体怎么使用可以参考 C++11 lambda 表达式解析 和 C++11 lambda 表达式

3.lua:

lua中的匿名函数被称为闭包(closure), 只能捕获和改变原值的克隆,不能改变原值(table除外)

4.python:

python中的匿名函数被称为lambda, 只能捕获克隆值,且不能改变他。


Java Lambda 表达式是一种简洁的方式来创建匿名函数。它们允许我们用一行代码替代传统的复杂的类实现,使得代码更易于编写、理解以及维护。Lambda表达式主要用于处理事件监听、回调等功能。 而闭包Closure),在Java这样的静态类型的环境中通常不是直接支持的特性,但在实际的应用场景中,我们可以构建类似闭包功能的行为。闭包是指有权访问另一个作用域变量的函数,这个作用域可以是局部作用域或者是外层作用域。在JavaScript等动态语言中,闭包是一个常见并且强大的特性,用于保存内部状态并提供对外部作用域变量的操作。 对于Java来说,通过结合`java.util.function`包下的接口(如`Function`, `Predicate`, `Consumer` 等)与Java 8引入的Lambda表达式,你可以实现类似于闭包的功能: 1. **局部Lambda**:将Lambda绑定到当前作用域的外部变量。虽然Java本身并不提供真正的闭包,但是当你在一个函数内部声明了一个Lambda,并引用了函数外部的作用域内的变量时,就相当于实现了“局部Lambda”,其行为类似于闭包。 ```java public class LocalClosureExample { private int x = 5; public void process() { int y = 10; Runnable myRunnable = () -> System.out.println("x = " + x + ", y = " + y); myRunnable.run(); } public static void main(String[] args) { new LocalClosureExample().process(); } } ``` 在这个例子中,即使函数体结束后局部变量`y`被销毁,但由于Lambda表达式`myRunnable`仍然存在,它能访问到外部作用域(`LocalClosureExample`)内的`x`变量,因此这在某种意义上实现了类似闭包的功能。 2. **通过成员变量实现类似闭包**:如果需要更复杂的状态保留,可以考虑将这些状态作为成员变量存储于一个类中,并在Lambda表达式中引用这些成员变量。 ```java public class ClosureExample { private int x = 5; public void process(ClosureCallback callback) { callback.doSomething(x); } public interface ClosureCallback { void doSomething(int value); } public static void main(String[] args) { ClosureExample example = new ClosureExample(); example.process((value) -> System.out.println("x = " + value)); } } ``` 在这个例子中,尽管没有直接使用Java中的闭包概念,通过接口`ClosureCallback`和Lambda表达式,我们可以传递一个包含对特定状态操作的函数(这里就是`doSomething`方法),使得函数可以在后续的操作中利用到外部类的实例成员变量`x`。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值