Python lambda 函数深度总结

今天我们来学习 Python 中的 lambda 函数,并探讨使用它的优点和局限性 Let's do it! ## 什么是 Python 中的 Lambda 函数 lambda 函数是一个匿名函数(即,没有名称定义),它可以接受任意数量的参数,但与普通函数不同,它只计算并返回一个表达式 Python 中的 lambda 函数使用以下语法表达: `lambda 参数:表达式` lambda 函数包括三个元素: - 关键字 lambda:与普通函数中 def 类似 - 参数:支持传递位置和关键字参数,与普通函数一样 - 正文:处理定参数的表达式 需要注意的是,普通函数不同,这里不需要用括号将 lambda 函数的参数括起来,如果 lambda 函数有两个或更多参数,我们用逗号列出它们 我们使用 lambda 函数只计算一个短表达式(理想情况下,单行)并且只计算一次,这意味着我们以后不会再复用这个函数。 通常来说我们会将 lambda 函数作为参数传递给高阶函数(接受其他函数作为参数的函数),例如 Python 内置函数,如 filter()、map() 或 reduce()等 ## Python 中的 Lambda 函数如何工作 让我们看一个简单的 lambda 函数示例: ```Python lambda x: x + 1 ``` Output: ```Text (x)> ``` 上面的 lambda 函数接受一个参数,将其递增 1,然后返回结果 它是以下带有 def 和 return 关键字的普通函数的更简单版本: ```Python def increment_by_one(x): return x + 1 ``` 到目前我们的 lambda 函数 `lambda x: x + 1` 只创建一个函数对象,不返回任何内容,这是因为我们没有为其参数 x 提供任何值(参数)。让我们先分配一个变量,将它传递给 lambda 函数,看看这次我们得到了什么: ```Python a = 2 print(lambda x: a + 1) ``` Output: ```Text at 0x00000250CB0A5820> ``` 我们的 lambda 函数没有像我们预期的那样返回 3,而是返回了函数对象本身及其内存位置,可以看出这不是调用 lambda 函数的正确方法。要将参数传递给 lambda 函数,执行它并返回结果,我们应该使用以下语法: ```Python (lambda x: x + 1)(2) ``` Output: ```Text 3 ``` 虽然我们的 lambda 函数的参数没有用括号括起来,但当我们调用它时,我们会在 lambda 函数的整个构造以及我们传递给它的参数周围添加括号 上面代码中要注意的另一件事是,使用 lambda 函数,我们可以在创建函数后立即执行该函数并接收结果。这就是所谓的立即调用函数执行(或 IIFE) 我们可以创建一个带有多个参数的 lambda 函数,在这种情况下,我们用逗号分隔函数定义中的参数。当我们执行这样一个 lambda 函数时,我们以相同的顺序列出相应的参数,并用逗号分隔它们: ```Python (lambda x, y, z: x + y + z)(3, 8, 1) ``` Output: ```Text 12 ``` 也可以使用 lambda 函数来执行条件操作。下面是一个简单 if-else 函数的 lambda 模拟: ```Python print((lambda x: x if(x > 10) else 10)(5)) print((lambda x: x if(x > 10) else 10)(12)) ``` Output: ```Text 10 12 ``` 如果存在多个条件(if-elif-...-else),我们必须嵌套它们: ```Python (lambda x: x * 10 if x > 10 else (x * 5 if x < 5 else x))(11) ``` Output: ```Text 110 ``` 但是上面的写法,又令代码变得难以阅读 在这种情况下,具有 if-elif-...-else 条件集的普通函数将是比 lambda 函数更好的选择。实际上,我们可以通过以下方式编写上面示例中的 lambda 函数: ```Python def check_conditions(x): if x > 10: return x * 10 elif x < 5: return x * 5 else: return x check_conditions(11) ``` Output: ```Text 110 ``` 尽管上面的函数比相应的 lambda 函数增加了更多行,但它更容易阅读 我们可以将 lambda 函数分配给一个变量,然后将该变量作为普通函数调用: ```Python increment = lambda x: x + 1 increment(2) ``` Output: ``` 3 ``` 但是根据 Python 代码的 PEP 8 样式规则,这是一种不好的做法 > 赋值语句的使用消除了 lambda 表达式相对于显式 def 语句所能提供的唯一好处(即,它可以嵌入到更大的表达式中) 因此如果我们确实需要存储一个函数以供进一步使用,我们最好定义一个等效的普通函数,而不是将 lambda 函数分配给变量 ## Lambda 函数在 Python 中的应用 ### 带有 filter() 函数的 Lambda Python 中的 filter() 函数需要两个参数: - 定义过滤条件的函数 - 函数在其上运行的可迭代对象 运行该函数,我们得到一个过滤器对象: ```Python lst = [33, 3, 22, 2, 11, 1] filter(lambda x: x > 10, lst) ``` Output: ```Text ``` 为了从过滤器对象中获取一个新的迭代器,并且原始迭代器中的所有项都满足预定义的条件,我们需要将过滤器对象传递给 Python 标准库的相应函数:list()、tuple()、set ()、frozenset() 或 sorted()(返回排序列表) 让我们过滤一个数字列表,只选择大于 10 的数字并返回一个按升序排序的列表: ```Python lst = [33, 3, 22, 2, 11, 1] sorted(filter(lambda x: x > 10, lst)) ``` Output: ```Text [11, 22, 33] ``` 我们不必创建与原始对象相同类型的新可迭代对象,此外我们可以将此操作的结果存储在一个变量中: ```Python lst = [33, 3, 22, 2, 11, 1] tpl = tuple(filter(lambda x: x > 10, lst)) tpl ``` Output: ```Text (33, 22, 11) ``` ### 带有 map() 函数的 Lambda 我们使用 Python 中的 map() 函数对可迭代的每个项目执行特定操作。它的语法与 filter() 相同:一个要执行的函数和一个该函数适用的可迭代对象。 map() 函数返回一个 map 对象,我们可以通过将该对象传递给相应的 Python 函数来从中获取一个新的迭代:list()、tuple()、set()、frozenset() 或 sorted() 与 filter() 函数一样,我们可以从 map 对象中提取与原始类型不同类型的可迭代对象,并将其分配给变量。 下面是使用 map() 函数将列表中的每个项目乘以 10 并将映射值作为分配给变量 tpl 的元组输出的示例: ```Python lst = [1, 2, 3, 4, 5] print(map(lambda x: x * 10, lst)) tpl = tuple(map(lambda x: x * 10, lst)) tpl ``` Output: ```Text (10, 20, 30, 40, 50) ``` map() 和 filter() 函数之间的一个重要区别是第一个函数总是返回与原始函数相同长度的迭代。因此由于 pandas Series 对象也是可迭代的,我们可以在 DataFrame 列上应用 map() 函数来创建一个新列: ```Python import pandas as pd df = pd.DataFrame({'col1': [1, 2, 3, 4, 5], 'col2': [0, 0, 0, 0, 0]}) print(df) df['col3'] = df['col1'].map(lambda x: x * 10) df ``` Output: ```Text col1 col2 0 1 0 1 2 0 2 3 0 3 4 0 4 5 0 col1 col2 col3 0 1 0 10 1 2 0 20 2 3 0 30 3 4 0 40 4 5 0 50 ``` 当然要在上述情况下获得相同的结果,也可以使用 apply() 函数: ```Python df['col3'] = df['col1'].apply(lambda x: x * 10) df ``` Output: ```Text col1 col2 col3 0 1 0 10 1 2 0 20 2 3 0 30 3 4 0 40 4 5 0 50 ``` 我们还可以根据某些条件为另一列创建一个新的 DataFrame 列,对于下面的代码,我们可以互换使用 map() 或 apply() 函数: ```Python df['col4'] = df['col3'].map(lambda x: 30 if x < 30 else x) df ``` Output: ```Text col1 col2 col3 col4 0 1 0 10 30 1 2 0 20 30 2 3 0 30 30 3 4 0 40 40 4 5 0 50 50 ``` ### 带有 reduce() 函数的 Lambda reduce() 函数与 functools Python 模块相关,它的工作方式如下: - 对可迭代对象的前两项进行操作并保存结果 - 对保存的结果和可迭代的下一项进行操作 - 以这种方式在值对上进行,直到所有项目使用可迭代的 该函数与前两个函数具有相同的两个参数:一个函数和一个可迭代对象。但是与前面的函数不同的是,这个函数不需要传递给任何其他函数,直接返回结果标量值: ```Python from functools import reduce lst = [1, 2, 3, 4, 5] reduce(lambda x, y: x + y, lst) ``` Output: ```Text 15 ``` 上面的代码展示了我们使用 reduce() 函数计算列表总和时的作用 需要注意的是,reduce() 函数总是需要一个带有两个参数的 lambda 函数,而且我们必须首先从 functools Python 模块中导入它 ## Python 中 Lambda 函数的优缺点 ### 优点 - 它是评估单个表达式的理想选择,应该只评估一次 - 它可以在定义后立即调用 - 与相应的普通语法相比,它的语法更紧凑 - 它可以作为参数传递给高阶函数,例如 filter()、map() 和 reduce() ### 缺点 - 它不能执行多个表达式 - 它很容易变得麻烦,可读性差,例如当它包括一个 if-elif-...-else 循环 - 它不能包含任何变量赋值(例如,lambda x: x=0 将抛出一个语法错误) - 我们不能为 lambda 函数提供文档字符串 ## 总结 总而言之,我们已经详细讨论了在 Python 中定义和使用 lambda 函数的许多方面: - lambda 函数与普通 Python 函数有何不同 - Python 中 lambda 函数的语法和剖析 - 何时使用 lambda 函数 - lambda 函数的工作原理 - 如何调用 lambda 函数 - 调用函数执行(IIFE)的定义 - 如何使用 lambda 函数执行条件操作,如何嵌套多个条件,以及为什么我们应该避免它 - 为什么我们应该避免将 lambda 函数分配给变量 - 如何将 lambda 函数与 filter() 函数一起使用 - 如何将 lambda 函数与 map() 函数一起使用 - 我们如何在 pandas DataFrame 中使用 - 带有传递给它的 lambda 函数的 map() 函数 - 以及在这种情况下使用的替代功能 - 如何将 lambda 函数与 reduce() 函数一起使用 - 在普通 Python 上使用 lambda 函数的优缺点 希望今天的讨论可以使 Python 中看似令人生畏的 lambda 函数概念更清晰、更易于应用,更希望小伙伴们能够喜欢,喜欢就点个赞吧! 本文由[mdnice](https://mdnice.com/?platform=6)多平台发布
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容