百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

10分钟学习函数式Python(python函数的基本用法)

off999 2024-09-26 16:18 23 浏览 0 评论

在这篇10分钟的文章中,您将学习Python中的函数式范型。您还将学习列表推导式。

目录

  1. 函数式范式
  2. Python的map函数是如何运行的
  3. Python中的lambda表达式
  4. Python中的reduce函数
  5. filter函数
  6. Python中的高阶函数
  7. 带有函数的部分应用
  8. 函数编程不是Python化
  9. 列表推导式
  10. 任何可迭代对象的推导式
  11. 结论

函数式范式

在命令式编程范式中,我们通过给计算机一个任务序列来执行任务,然后计算机会执行这些任务。在执行它们时,计算机可以改变状态。例如,我们设A = 5,然后改变A的值。因为我们的A是变量,所以它内部的值是变化的。

在函数式编程范式中,我们不告诉计算机去做什么,而是告诉它是什么东西。一个数的最大公约数是什么,等等。

变量不会变化。一旦我们设置了一个变量,它就会永远保持这种状态。因此,函数在函数式范型中没有副作用。副作用就是函数改变了它外部的东西。让我们来看一个例子:

输出是5。在函数式范型中,改变变量是一个很大的禁忌,而让函数影响它们范围之外的东西也是一个大大的禁忌。函数唯一能做的就是计算某些东西并返回它。

现在您可能会想“没有变量,就没有副作用?为什么这很好?”问得好,读这篇文章的古怪陌生人。

如果一个函数使用相同的参数被调用两次,那么它肯定会返回相同的结果。如果您学过数学函数,您就会喜欢这一点。我们称之为函数的引用透明性。由于函数没有副作用,如果我们构建一个计算程序,我们就可以加快该程序的速度。如果程序知道func(2)等于3,我们可以将其存储在一个表中。这将防止程序在我们已经知道答案的情况下去运行相同的函数。

通常,在函数式编程中,我们不使用循环,我们使用递归。递归是一个数学概念,它意味着“自食其力”。对于递归函数,该函数将自己作为一个子函数进行调用。下面是Python中递归函数的一个很好的例子:

一些编程语言也很懒。这意味着他们直到最后一秒才开始计算或做任何事情。如果我们编写一些代码来执行2 + 2,一个函数式程序只会在我们需要使用结果时才会计算这个结果。我们很快就会探讨Python中的惰性。

Python的map函数是如何运行的

为了理解映射,让我们首先看看什么是可迭代对象。一个可迭代对象是我们可以迭代的任何东西。这些是列表或数组,但是Python有许多不同的可迭代对象。我们甚至可以通过实现魔术方法来创建我们自己的可迭代对象。一个魔术方法就像一个API,它可以帮助我们的对象变得更Python化。要使一个对象成为一个可迭代对象,我们需要实现2个魔术方法:

第一个魔术方法是__iter__,或者叫特殊iter(双下划线)方法,它会返回迭代对象,我们通常在循环开始时使用它。特殊next方法,__next__,会返回下一个对象是什么。

让我们来看看这个:

这将输出:

在Python中,迭代器是一个对象,它只有一个简单的__iter__魔术方法。这意味着我们可以访问该对象中的位置,但不能遍历该对象。有些对象有魔术方法__next__,但没有__iter__魔术方法,如sets(将在本文后面讨论)。对于本文,我们将假设我们接触的所有东西都是一个可迭代的对象。

现在我们知道了什么是可迭代对象,让我们回到map函数。

map函数允许我们将一个函数应用到一个可迭代对象中的每个项。我们希望将一个函数应用到一个列表中的每个项,但是要知道这对大多数可迭代对象来说都是可行的。map的语法接受两个输入,即要应用的函数和可迭代的对象。

假设我们有一个像这样的数字列表:

我们相对每个数字进行平方,我们可以像这样写代码:

函数式Python是惰性的。如果我们不包含list(),函数将存储该可迭代对象的定义,而不是列表本身。我们需要告诉Python“把这个转换成一个列表”,以便我们使用它。

在Python中突然从非惰性求值变成惰性求值是很奇怪的。如果您更多地以函数式思维而不是命令式思维进行思考,您就会习惯它。

现在写一个像square(num)这样的普通函数就很好了,但是它看起来不够好。我们必须定义一个完整的函数才可以在一个映射中使用它吗?好吧,我们可以使用lambda(匿名)函数在map中定义一个函数。

Python中的lambda表达式

Lambda函数是一个只有一行代码的函数,适用于短期内使用。我们经常将它们随同高阶函数,如filter、map和reduce函数,一起使用。这个lambda表达式会对传给它的数字进行平方:

现在我们来运行这个函数:

我听到您在说:“Brandon,参数在哪里?这是什么鬼东西?它看起来一点也不像一个函数?”

嗯,这确实很令人困惑,但我可以解释它。我们将某个东西赋值给变量square。这部分:

告诉Python这是一个lambda函数,输入被称为x。冒号之后的任何东西都是我们对输入所执行的操作,它返回的就是这些操作的结果。

为了将我们的平方程序简化成一行,我们可以这样做:

在一个lambda表达式中,所有的参数都在左边,而我们要用它们做的事情都在右边。没人能否认,这有点乱。编写只有其他函数式程序员才能阅读的代码是一种乐趣。另外,将一个函数转换成一行程序是非常酷的事情。

Python中的reduce函数

reduce是一个函数,它将给定的函数应用于一个可迭代对象并返回一个东西。通常我们会在一个列表上进行计算,将其缩减至一个数字。Reduce看起来是这样的:

我们可以(通常也会)使用lambda表达式作为函数。

列表的乘积是每一个数字相乘。编写的程序是这样:

但是使用reduce我们可以这样写:

我们得到了相同的乘积。代码更短,并且具有函数式编程的知识,因此更简洁。

fileter函数

filter函数接受一个iterable并过滤掉我们不希望存在于该iterable中的所有东西。

filter接受一个函数和一个列表。它将该函数应用于列表中的每一项,如果该函数返回True,则不执行任何操作。如果该函数返回False,它会从该列表中删除该项。

语法如下:

让我们看一个小例子,没有filter,我们会这样写:

使用filter, 这就变成了:

Python中的高阶函数

高阶函数可以将函数作为参数并返回函数。一个例子是:

或者第二个定义,return functions,的一个简单例子是:

高阶函数使非变化变量更容易处理。如果我们所做的只是在一系列函数中传递数据,那么我们就不需要在任何地方存储变量。

Python中的所有函数都是一级对象。当一个对象具有以下特性中的一个或多个时,我们将其定义为一级对象:

  • 在运行时被创建
  • 可以被赋值给一个变量或一个数据结构中的元素
  • 作为参数被传递给函数
  • 作为函数的结果被返回

因此Python中的所有函数都是一级函数,可以作为高阶函数使用。

带函数的部分应用

部分应用(也称为闭包)很奇怪,但是也很酷。我们可以调用一个函数而不提供它需要的所有参数。我们来在一个例子中看一下这一点。我们想要创建一个函数,它接受两个参数,一个基数和一个指数,然后返回基数的指数次方,就像这样:

现在我们想要一个专用的平方函数,来使用power数求出一个数的平方:

这是可行的,但如果我们想要一个立方函数呢?或者一个4次方函数呢?我们能一直写下去吗?嗯,我们可以。但是程序员是很懒的。如果我们重复地做同一件事时,这是一个信号,表明有一种更快的方法来加快做这些事情的速度,那将允许我们不再重复地做这些事情。我们可以在这里使用部分应用。让我们看一个使用了一个部分应用的平方函数的例子:

这难道不酷吗?我们可以通过告诉Python第二个参数是什么来只使用一个参数调用需要两个参数的函数。

我们还可以使用一个循环,来生成一个幂函数,其运行范围可以从立方到1000次幂。

函数式编程不是Python化

您可能已经注意到了,我们在函数式编程中想要做的很多事情都是围绕列表进行的。除了reduce函数和部分应用外,我们所看到的所有函数都会生成列表。Guido (Python的发明者)不喜欢Python中的函数式的东西,因为Python已经有了自己的生成列表的方法。

如果我们在一个Python IDLE会话中输入“import this”,我们会得到:

这就是Python之禅。这是一首关于某些东西Python化意味着什么的诗。这里我们要涉及的部分是: 应该有一种——最好是只有一种——显而易见的方法来做到它。

在Python中,map 与 filter可以做与列表表达式(接下来讨论)相同的事情。这打破了Python之禅中的一条规则,因此函数式编程的这些部分不是“Python式的”。

另一个话题是Lambda。在Python中,lambda函数是一个普通函数。Lambda是语法糖。这两个是等价的:

一个普通函数可以做lambda函数所能做的所有事情,但反过来却不行。一个lambda函数不能完成一个普通函数所能完成的所有工作。

这是一个关于函数式编程为什么不能很好地适应整个Python生态系统的简短讨论。您可能已经注意到我之前提到过列表推导式,我们现在将讨论它们。

列表推导式

之前我提到过,我们可以用列表推导式完成我们可以用map或filter所做的任何事情。这是我们要学习的关于它们的部分。

列表推导式是在Python中生成列表的一种方式。其语法是:

让我们对一个列表中的每个数字进行平方,并以此作为一个例子:

好吧,这样我们就可以看到我们如何将一个函数应用到列表中的每一项。我们如何来应用一个filter函数呢?好吧,看看这段之前的代码:

我们可以像这样来把它转换成一个列表推导式:

列表推导式支持像这样的if语句。我们不需要再应用很多个函数来得到我们想要的东西了。如果我们试图创造使用列表的机会,那么使用列表推导式可能会更清晰、更容易一些。

如果我们想要对列表中所有小于0的数进行平方呢?那么,使用lambda、map和filter,我们会这样写:

这有点冗长而复杂。使用一个列表推导式,它就会变成这样:

列表推导式只对列表有好处。map和filter作用于任何可迭代对象之上,那是怎么回事呢?我们可以对遇到的任何可迭代对象使用任何推导式。

任何可迭代对象的推导式

我们可以使用一个推导式来生成任何可迭代对象。因为我们使用的是Python 2.7,所以,我们甚至可以生成一个字典(hashmap)。

如果它是一个可迭代对象,我们可以生成它。我们来看集合的最后一个例子。如果您不知道集合是什么,请查看我写的另一篇文章(https://skerritt.blog/a-primer-on-set-theory/ )。其中的TL;DR(集合定义)是:

  • 集合是元素的列表,该列表中没有重复的元素
  • 集合的顺序无关紧要。

您可能会注意到,集合具有与字典相同的花括号。Python是很聪明的。它会根据我们是否为字典提供额外的值来判断我们写的是一个字典推导式还是一个集合推导式。如果您想了解更多关于推导式的内容,请查看这个可视化指南。

结论

函数式编程是漂亮而纯粹的。函数式代码可以是简洁的,但也可能是混乱的。您应该根据需要去使用它。

英文原文:https://skerritt.blog/learn-functional-python-in-10-minutes/

译者:测试

相关推荐

Linux 网络协议栈_linux网络协议栈

前言;更多学习资料(包含视频、技术学习路线图谱、文档等)后台私信《资料》免费领取技术点包含了C/C++,Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,Z...

揭秘 BPF map 前生今世_bpfdm

1.前言众所周知,map可用于内核BPF程序和用户应用程序之间实现双向的数据交换,为BPF技术中的重要基础数据结构。在BPF程序中可以通过声明structbpf_map_def...

教你简单 提取fmpeg 视频,音频,字幕 方法

ffmpeg提取视频,音频,字幕方法(HowtoExtractVideo,Audio,SubtitlefromOriginalVideo?)1.提取视频(ExtractVi...

Linux内核原理到代码详解《内核视频教程》

Linux内核原理-进程入门进程进程不仅仅是一段可执行程序的代码,通常进程还包括其他资源,比如打开的文件,挂起的信号,内核内部的数据结构,处理器状态,内存地址空间,或多个执行线程,存放全局变量的数据段...

Linux C Socket UDP编程详解及实例分享

1、UDP网络编程主要流程UDP协议的程序设计框架,客户端和服务器之间的差别在于服务器必须使用bind()函数来绑定侦听的本地UDP端口,而客户端则可以不进行绑定,直接发送到服务器地址的某个端口地址。...

libevent源码分析之bufferevent使用详解

libevent的bufferevent在event的基础上自己维护了一个buffer,这样的话,就不需要再自己管理一个buffer了。先看看structbufferevent这个结构体struct...

一次解决Linux内核内存泄漏实战全过程

什么是内存泄漏:程序向系统申请内存,使用完不需要之后,不释放内存还给系统回收,造成申请的内存被浪费.发现系统中内存使用量随着时间的流逝,消耗的越来越多,例如下图所示:接下来的排查思路是:1.监控系统中...

彻底搞清楚内存泄漏的原因,如何避免内存泄漏,如何定位内存泄漏

作为C/C++开发人员,内存泄漏是最容易遇到的问题之一,这是由C/C++语言的特性引起的。C/C++语言与其他语言不同,需要开发者去申请和释放内存,即需要开发者去管理内存,如果内存使用不当,就容易造成...

linux网络编程常见API详解_linux网络编程视频教程

Linux网络编程API函数初步剖析今天我们来分析一下前几篇博文中提到的网络编程中几个核心的API,探究一下当我们调用每个API时,内核中具体做了哪些准备和初始化工作。1、socket(family...

Linux下C++访问web—使用libcurl库调用http接口发送解析json数据

一、背景这两天由于一些原因研究了研究如何在客户端C++代码中调用web服务端接口,需要访问url,并传入json数据,拿到返回值,并解析。 现在的情形是远程服务端的接口参数和返回类型都是json的字符...

平衡感知调节:“系统如人” 视角下的架构设计与业务稳定之道

在今天这个到处都是数字化的时代,系统可不是一堆冷冰冰的代码。它就像一个活生生的“数字人”,没了它,业务根本转不起来。总说“技术要为业务服务”,但实际操作起来问题不少:系统怎么才能快速响应业务需求?...

谈谈分布式文件系统下的本地缓存_什么是分布式文件存储

在分布式文件系统中,为了提高系统的性能,常常会引入不同类型的缓存存储系统(算法优化所带来的的效果可能远远不如缓存带来的优化效果)。在软件中缓存存储系统一般可分为了两类:一、分布式缓存,例如:Memca...

进程间通信之信号量semaphore--linux内核剖析

什么是信号量信号量的使用主要是用来保护共享资源,使得资源在一个时刻只有一个进程(线程)所拥有。信号量的值为正的时候,说明它空闲。所测试的线程可以锁定而使用它。若为0,说明它被占用,测试的线程要进入睡眠...

Qt编写推流程序/支持webrtc265/从此不用再转码/打开新世界的大门

一、前言在推流领域,尤其是监控行业,现在主流设备基本上都是265格式的视频流,想要在网页上直接显示监控流,之前的方案是,要么转成hls,要么魔改支持265格式的flv,要么265转成264,如果要追求...

30 分钟搞定 SpringBoot 视频推拉流!实战避坑指南

30分钟搞定SpringBoot视频推拉流!实战避坑指南在音视频开发领域,SpringBoot凭借其快速开发特性,成为很多开发者实现视频推拉流功能的首选框架。但实际开发中,从环境搭建到流处理优...

取消回复欢迎 发表评论: