【深度学习】 Python 和 NumPy 系列教程(七):Python函数

2023-11-12

目录

一、前言

二、实验环境

三、Python函数基础

1. 定义函数

2. 参数传递

3. 函数调用

4. 返回值

5. 函数文档字符串

四、将函数存储在模块中

1. 创建模块

2. 导入模块

a. import 模块名

b. from 模块名 import 函数名

c. from 模块名 import *

五、多种形式的函数

1. 普通函数

2. 匿名函数(Lambda函数)

3. 内置函数

4. 递归函数

a. 递归概念

b. 递归条件

5. 高阶函数

6. 装饰器函数

7. 生成器函数

8. 异步函数

9. 偏函数


一、前言

        Python是一种高级编程语言,由Guido van Rossum于1991年创建。它以简洁、易读的语法而闻名,并且具有强大的功能和广泛的应用领域。Python具有丰富的标准库和第三方库,可以用于开发各种类型的应用程序,包括Web开发、数据分析、人工智能、科学计算、自动化脚本等。

        Python本身是一种伟大的通用编程语言,在一些流行的库(numpy,scipy,matplotlib)的帮助下,成为了科学计算的强大环境。本系列将介绍Python编程语言和使用Python进行科学计算的方法,主要包含以下内容:

  • Python:基本数据类型、容器(列表、元组、集合、字典)、函数、类
  • Numpy:数组、数组索引、数据类型、数组数学、广播
  • Matplotlib:绘图,子图,图像
  • IPython:创建笔记本,典型工作流程

二、实验环境

numpy 1.21.6
python 3.7.16
  • 运行下述命令检查Python版本
 python --version 
  • 运行下述代码检查Python、NumPy版本
import sys
import numpy as np

print("Python 版本:", sys.version)
print("NumPy 版本:", np.__version__)

三、Python函数基础

        Python函数指的是一段可重复使用的代码块,用于执行特定的任务。函数接受输入参数(可选)并返回一个输出结果(也可选)。Python函数有以下几个关键特点:

1. 定义函数

        使用关键字def来定义函数。函数定义包括函数名、参数列表和函数体。

def add_numbers(a, b):
    sum = a + b
    return sum

        上述代码定义了一个名为add_numbers的函数,它接受两个参数ab,并返回它们的和。

2. 参数传递

        函数可以接受零个或多个参数。参数可以是必需的(必须提供)或可选的(可以省略)。函数在调用时通过参数来接收输入值。

def greet(name):
    print("Hello, " + name + "!")

        上述代码定义了一个名为greet的函数,它接受一个名为name的参数,并打印出问候语。

3. 函数调用

        要调用函数,可以使用函数名和相应的参数列表。

result = add_numbers(3, 4)
print(result)

        上述代码调用了add_numbers函数,并将返回的结果赋值给result变量,然后打印出结果。

4. 返回值

        函数可以使用return语句返回一个值。返回值可以是任意类型的对象,如数字、字符串、列表等。

def multiply_numbers(a, b):
    product = a * b
    return product

        上述代码定义了一个名为multiply_numbers的函数,它接受两个参数ab,并返回它们的乘积。

5. 函数文档字符串

        为了方便其他开发人员理解函数的用途和使用方法,可以在函数内部使用文档字符串(docstring)进行注释。文档字符串是位于函数定义之后的字符串,可以通过help()函数或.__doc__属性来查看。

def add_numbers(a, b):
    """
    This function adds two numbers and returns the result.
    """
    sum = a + b
    return sum


help(add_numbers)

print(add_numbers.__doc__)

        

四、将函数存储在模块中

1. 创建模块

        创建一个新的Python文件,并命名为希望的模块名(例如my_module.py)。在该文件中,定义函数和其他相关代码。

def add_numbers(a, b):
    return a + b

def multiply_numbers(a, b):
    return a * b

# 其他函数和代码...

        保存文件并将其放在Python解释器可以访问的位置。通常,可以将模块文件与调用它的代码文件放在同一个目录中。

2. 导入模块

        将函数存储在模块中可以提高代码的组织性和可重用性。模块是一种将相关功能封装在一起的方式,可以在项目中的多个文件中使用它们,并且可以与其他开发人员共享和重用。

a. import 模块名

        在另一个Python脚本中,通过使用import语句导入创建的模块。

import my_module

result = my_module.add_numbers(3, 5)
print(result)  # 输出:8

result = my_module.multiply_numbers(2, 4)
print(result)  # 输出:8

        通过import语句导入模块后,就可以使用模块中定义的函数和其他代码。可以通过模块名.函数名的方式来调用模块中的函数。

b. from 模块名 import 函数名

        使用from 模块名 import 函数名的形式,可以直接使用函数名调用,而无需使用模块名作为前缀。    

from my_module import add_numbers

result = add_numbers(3, 5)
print(result)  # 输出:8

        这种方式可以选择性地导入模块中的特定函数或变量,以便更方便地使用它们。

c. from 模块名 import *

        使用from 模块名 import *的方式可以导入模块中的所有函数和变量。这种导入方式将模块中所有的公开(没有以下划线开头的)函数和变量都导入到当前命名空间中。

        然而,建议尽量避免使用from 模块名 import *的方式导入模块,特别是在大型项目中。这是因为这种方式可能导致命名空间污染和命名冲突的问题。当导入的模块中有多个函数或变量与当前命名空间中的名称相同时,会发生命名冲突,导致不可预料的行为。

        相反,推荐使用显式导入的方式,即使用from 模块名 import 函数名import 模块名的形式。这样可以明确指定要导入的函数或模块,并且在使用时可以清楚地知道其来源。

        如果确实需要导入模块中的所有函数和变量,可以使用import 模块名的方式导入整个模块,并在使用时通过模块名.函数名的方式来调用它们。这样可以避免命名冲突,并且更清晰地表达代码的意图。

五、多种形式的函数

1. 普通函数

        普通函数是最常见的函数形式,由def关键字定义,可以接受参数并返回值。

def add_numbers(a, b):
    return a + b

result = add_numbers(3, 5)
print(result)  # 输出:8

2. 匿名函数(Lambda函数)

        匿名函数(lambda函数)是一种没有函数名的简单函数形式。它通常用于需要一次性定义并使用的简单函数。匿名函数使用lambda关键字定义,并可以包含一个或多个参数和一个表达式作为函数体。以下是一个使用匿名函数计算两个数的和的示例:

add_numbers = lambda a, b: a + b
result = add_numbers(3, 5)
print(result)  # 输出:8

3. 内置函数

        Python提供了许多内置函数,这些函数是Python解释器提供的预定义函数,可以直接使用。这些内置函数包括len()print()range()type()等等,用于执行各种常见的操作。以下是一些常用的内置函数的示例:

# 获取字符串长度
length = len("Hello, world!")
print(length)  # 输出:13

# 打印文本
print("Hello, world!")

# 生成整数序列
numbers = list(range(1, 6))
print(numbers)  # 输出:[1, 2, 3, 4, 5]

# 获取对象类型
print(type(numbers))  # 输出:<class 'list'>

4. 递归函数

a. 递归概念

        函数递归是指函数在其函数体内调用自身的过程。递归函数通常包含两个部分:基本情况和递归情况。

  • 基本情况是指函数停止递归的条件。当满足基本情况时,递归函数不再调用自身,而是返回一个特定的值或执行其他操作。
  • 递归情况是指函数继续递归调用自身的条件。在递归情况下,函数会通过传递不同的参数值来解决更小规模的问题。通过不断缩小问题的规模,最终达到基本情况,从而结束递归。

b. 递归条件

递归函数需要满足以下两个重要条件:

  • 基本情况:必须存在一个或多个基本情况,用于终止递归并返回特定的值或执行特定的操作。

  • 收敛性:递归调用必须朝着基本情况逼近。也就是说,在每次递归调用中,问题的规模都应该比上一次递归调用要小,最终达到基本情况。

        如果递归函数没有正确定义基本情况或无法收敛,就会导致无限递归,最终导致栈溢出或程序崩溃。递归函数在某些情况下可以提供一种简洁、优雅的解决方案。然而,递归的执行过程相对于迭代(循环)来说更消耗内存和时间,因此在使用递归时需要注意问题规模和性能。

        下面是一个经典的递归函数示例,计算一个正整数的阶乘:

def factorial(n):
    if n == 0:
        return 1
    else:
        return n * factorial(n - 1)

result = factorial(5)
print(result)  # 输出:120

        在上述代码中,当n等于0时,递归函数返回1作为基本情况。否则,递归函数计算nfactorial(n - 1)的乘积作为递归情况。

5. 高阶函数

        高阶函数是一种可以接受函数作为参数或返回函数的函数形式。在Python中,函数是一等公民,因此可以像任何其他对象一样进行传递和操作。高阶函数可以用于实现函数的组合、过滤、映射等操作。例如,map()filter()是常见的高阶函数,用于对可迭代对象进行映射和过滤操作。以下是一个使用map()filter()的示例:

numbers = [1, 2, 3, 4, 5]

# 使用map()函数将每个数平方
squared_numbers = list(map(lambda x: x**2, numbers))
print(squared_numbers)  # 输出:[1, 4, 9, 16, 25]

# 使用filter()函数过滤出偶数
even_numbers = list(filter(lambda x: x % 2 == 0, numbers))
print(even_numbers)  # 输出:[2, 4]

        在上面的例子中,map()函数将每个数平方,并使用lambda x: x**2作为映射函数。filter()函数过滤出偶数,并使用lambda x: x % 2 == 0作为过滤函数。

6. 装饰器函数

  • 装饰器函数是一种特殊的函数,用于修改其他函数的行为或功能。
    • 装饰器函数通常接受一个函数作为输入,并返回一个新的函数作为输出。
    • 装饰器函数可以在不修改原始函数代码的情况下,通过添加额外的功能来扩展函数的行为。
  • 下面是一个简单的装饰器函数,用于在函数调用前后打印日志:
def logger(func):
    def wrapper(*args, **kwargs):
        print("Calling function:", func.__name__)
        result = func(*args, **kwargs)
        print("Function", func.__name__, "finished execution")
        return result
    return wrapper

@logger
def add_numbers(a, b):
    return a + b

result = add_numbers(3, 5)
print(result)  # 输出:8

        在上面的例子中,logger装饰器函数接受一个函数作为输入,并返回一个新的函数wrapperwrapper函数在调用被装饰的函数之前和之后打印日志信息。

7. 生成器函数

  • 生成器函数是一种特殊的函数,可以用于定义生成器。
    • 生成器是一种特殊的迭代器,可以按需生成值,而不是一次性生成所有值。
    •  生成器函数使用yield关键字来定义生成器的每个元素。
    • 每次调用生成器的next()函数或使用for循环迭代时,生成器函数会从上次暂停的位置继续执行,并生成下一个值。
    • 这种按需生成值的方式可以提高性能和节省内存。
  • 下面是一个生成斐波那契数列的生成器函数:
def fibonacci_generator():
    a, b = 0, 1
    while True:
        yield a
        a, b = b, a + b

fib = fibonacci_generator()
print(next(fib))  # 输出:0
print(next(fib))  # 输出:1
print(next(fib))  # 输出:1
print(next(fib))  # 输出:2
# ...

# 使用for循环打印斐波那契数列的前十个数
fib = fibonacci_generator()
for _ in range(10):
    print(next(fib))

        在上面的例子中,fibonacci_generator生成器函数使用yield关键字定义了一个生成斐波那契数列的生成器。每次调用next(fib)时,生成器会生成下一个斐波那契数列的值。

8. 异步函数

        异步函数是一种用于异步编程的函数形式,可以使用async关键字定义。异步函数通常与await关键字一起使用,用于处理异步操作,例如网络请求、文件读写等。异步函数能够提高程序的并发性能和响应性,允许在等待某些操作完成时执行其他任务。以下是一个简单的异步函数的示例:

import asyncio

async def greet(name):
    print("Hello, " + name)
    await asyncio.sleep(1)
    print("Goodbye, " + name)

asyncio.run(greet("Alice"))

        在上面的例子中,greet异步函数使用await关键字等待异步操作asyncio.sleep(1)完成。在等待期间,可以执行其他任务。这样的异步函数可以在需要等待I/O操作的情况下提高程序的性能。

9. 偏函数

        偏函数是一种固定函数部分参数的函数形式。它通过使用functools.partial()函数来创建一个新的函数,该函数固定了原始函数的部分参数。偏函数可以用于简化函数调用,减少重复代码。以下是一个使用偏函数的示例:

import functools

def power(base, exponent):
    return base ** exponent

square = functools.partial(power, exponent=2)
cube = functools.partial(power, exponent=3)

print(square(4))  # 输出:16,等同于 power(4, 2)
print(cube(4))  # 输出:64,等同于 power(4, 3)

        在上面的例子中,functools.partial()函数创建了两个新的偏函数squarecube,它们是power函数的特定版本,其中exponent参数被固定为2和3。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【深度学习】 Python 和 NumPy 系列教程(七):Python函数 的相关文章

  • 我怎样才能更多地了解Python的内部原理? [关闭]

    Closed 这个问题正在寻求书籍 工具 软件库等的推荐 不满足堆栈溢出指南 help closed questions 目前不接受答案 我使用Python编程已经有半年多了 我对Python内部更感兴趣 而不是使用Python开发应用程序
  • 如何在序列化器创建方法中获取 URL Id?

    我有以下网址 url r member P
  • 如何正确地将 MIDI 刻度转换为毫秒?

    我正在尝试将 MIDI 刻度 增量时间转换为毫秒 并且已经找到了一些有用的资源 MIDI Delta 时间刻度到秒 http www lastrayofhope co uk 2009 12 23 midi delta time ticks
  • python 模拟第三方模块

    我正在尝试测试一些处理推文的类 我使用 Sixohsix twitter 来处理 Twitter API 我有一个类充当 Twitter 类的外观 我的想法是模拟实际的 Sixohsix 类 通过随机生成新推文或从数据库检索它们来模拟推文的
  • 通过列表理解压平列表列表

    我正在尝试使用 python 中的列表理解来展平列表 我的清单有点像 1 2 3 4 5 6 7 8 只是为了打印这个列表列表中的单个项目 我编写了这个函数 def flat listoflist for item in listoflis
  • 如何在 pytest 中将单元测试和集成测试分开

    根据维基百科 https en wikipedia org wiki Unit testing Description和各种articles https techbeacon com devops 6 best practices inte
  • 如何计算numpy数组中元素的频率?

    我有一个 3 D numpy 数组 其中包含重复的元素 counterTraj shape 13530 1 1 例如 counterTraj 包含这样的元素 我只显示了几个元素 array 136 129 130 103 102 101 我
  • 为什么Python的curses中escape键有延迟?

    In the Python curses module I have observed that there is a roughly 1 second delay between pressing the esc key and getc
  • 切片 Dataframe 时出现 KeyError

    我的代码如下所示 d pd read csv Collector Output csv df pd DataFrame data d dfa df copy dfa dfa rename columns OBJECTID Object ID
  • 在Python中调整图像大小

    我有一张尺寸为 288 352 的图像 我想将其大小调整为 160 240 我尝试了以下代码 im imread abc png img im resize 160 240 Image ANTIALIAS 但它给出了一个错误TypeErro
  • 从 python 发起 SSH 隧道时出现问题

    目标是在卫星服务器和集中式注册数据库之间建立 n 个 ssh 隧道 我已经在我的服务器之间设置了公钥身份验证 因此它们只需直接登录而无需密码提示 怎么办 我试过帕拉米科 它看起来不错 但仅仅建立一个基本的隧道就变得相当复杂 尽管代码示例将受
  • 将 JSON 对象传递给带有请求的 url

    所以 我想利用 Kenneth 的优秀请求模块 https github com kennethreitz requests 在尝试使用时偶然发现了这个问题自由库API http wiki freebase com wiki API 基本上
  • 如何在 python 中没有 csv.reader 迭代器的情况下解析单行 csv 字符串?

    我有一个 CSV 文件 需要重新排列和重新编码 我想跑 line line decode windows 1250 encode utf 8 在由 CSV 读取器解析和分割之前的每一行 或者我想自己迭代行 运行重新编码 并仅使用单行解析表单
  • Python 将日志滚动到变量

    我有一个使用多线程并在服务器后台运行的应用程序 为了无需登录服务器即可监控应用程序 我决定包括Bottle http bottlepy org为了响应一些HTTP端点并报告状态 执行远程关闭等 我还想添加一种查阅日志文件的方法 我可以使用以
  • 使用 PyTorch 分布式 NCCL 连接失败

    我正在尝试使用 torch distributed 将 PyTorch 张量从一台机器发送到另一台机器 dist init process group 函数正常工作 但是 dist broadcast 函数中出现连接失败 这是我在节点 0
  • 当鼠标悬停在上面时,intellisense vscode 不显示参数或文档

    我正在尝试将整个工作流程从 Eclipse 和 Jupyter Notebook 迁移到 VS Code 我安装了 python 扩展 它应该带有 Intellisense 但它只是部分更糟糕 我在输入句点后收到建议 但当将鼠标悬停在其上方
  • 具有自定义值的 Django 管理外键下拉列表

    我有 3 个 Django 模型 class Test models Model pass class Page models Model test models ForeignKey Test class Question model M
  • 检查字典键是否有空值

    我有以下字典 dict1 city name yass region zipcode phone address tehsil planet mars 我正在尝试创建一个基于 dict1 的新字典 但是 它不会包含带有空字符串的键 它不会包
  • 从 Twitter API 2.0 获取 user.fields 时出现问题

    我想从 Twitter API 2 0 端点加载推文 并尝试获取标准字段 作者 文本 和一些扩展字段 尤其是 用户 字段 端点和参数的定义工作没有错误 在生成的 json 中 我只找到标准字段 但没有找到所需的 user fields 用户
  • 您可以使用关键字参数而不提供默认值吗?

    我习惯于在 Python 中使用这样的函数 方法定义 def my function arg1 None arg2 default do stuff here 如果我不供应arg1 or arg2 那么默认值None or default

随机推荐

  • C++ 分数的加减乘除(运算符重载)

    文章目录 一 题目描述 二 输入与输出 1 输入 2 输出 三 参考代码 一 题目描述 Fraction类的基本形式如下 定义Fraction类 class Fraction private int fz fm int commonDivi
  • 奇异值分解的反变换matlab程序,奇异值分解(SVD)基础概念及MATLAB仿真

    奇异值分解 SVD 基础概念及MATLAB仿真 奇异值分解 SVD 基础概念及MATLAB仿真 奇异值分解 singular value decomposition 简称SVD 不仅广泛应用于机器学习领域 也在控制理论中有着广泛的应用 本文
  • springboot 启动异常Input length = 1解决

    问题原因是yml配置文件内容引起 并且yml内容没问题 空格也没问题 而是编码问题引起 在idea中打开settings gt Editor gt File Econdings 设置yml为UTF 8 如果还是启动失败 解决方法 将yml内
  • 一个合格(优秀)的前端都应该阅读这些文章

    的确 有些标题党了 起因是微信群里 有哥们问我 你是怎么学习前端的呢 能不能共享一下学习方法 一句话也挺触动我的 我真的不算是什么大佬 对于学习前端知识 我也不能说是掌握了什么捷径 当然 我个人的学习方法这篇文章已经在写了 预计这周末会在我
  • The overload Pattern

    最近研究有关C 17的书和博客文章时 偶然发现了这种访问模式std variant 使用overload模式 您可以 访问 提供单独的lambda include
  • 如何查看端口占用

    1 Win R 打开 运行 在 运行 中输入cmd 然后按电脑的回车进入命令行 2 输入命令 netstat ano 即可查出所有的端口占用信息 3 netstat ano findstr 端口 可查看具体的端口占用信息 4 查看有效会话
  • 关于B/S结构系统的会话机制(session机制)

    关于B S结构系统的会话机制 session机制 什么是会话 用户打开浏览器 进行一系列操作 最后关闭浏览器 这个整个过程叫做一次会话 真正意义上 会话是session对象的创建到结束的过程 一个session对应一个会话 在服务器端有一个
  • 虚拟机如何进行备份和迁移?

    文章目录 一 虚拟机的备份 方式一 使用VMvare自带的克隆功能 方式二 手动复制 二 虚拟机的迁移 本文以Ubuntu18 04为例 使用的工具是VMware Workstation 16 Pro 即VMware 最新版 Ubuntu1
  • 谷歌直链下载云盘数据集,使用谷歌云盘生成直接下载的url,示例: NeRFStudio-nerfacto默认训练数据集poster下载,nerfstudio 数据集下载链接

    1 如何生成谷歌直链下载云盘数据集的url 谷歌云盘 NeRFStudio nerfacto默认训练数据集poster下载链接 https drive google com uc export download confirm id 1dm
  • 差分与微分

    差分与微分 差分和微分都是数学中的一些基本概念 它们在数学 物理学 经济学和其他领域中都有广泛的应用 1 差分 差分可以理解为函数值之间的差别 特别地 如果一个函数 f f f 在两个相邻点 x 0
  • 【Qt教程】2.6 - Qt5 自定义控件封装

    自定义控件是将原有控件组合 封装 并填入自定义功能 留出适当的函数接口 方便其他控件的调用 设计窗口时 可以将控件拖入 方便模块化设计 GitHub代码仓库 Qt学习例程 1 建立工程 新建自定义控件 创建widget工程 新建控件 Add
  • 青龙面板打不开解决办法

    原因是cdn 自己修改一下就可以了 进入文件夹etc找到hosts etchosts 把这二个添加 104 16 88 20 cdn js delivr net 直接复制在最后一行 就可以解决问题了 或者 1 登陆linux 2 进入青龙容
  • alter database open resetlogs

    问题解决 先要弄清楚alter database open resetlogs是什么意思 为什么要用resetlogs打开数据库 这个命令发出后oracle都做了什么 alter database open resetlogs是要打开数据时
  • pytorch——torch.squeeze() 和torch.unsqueeze()的用法

    torch squeeze torch squeeze 这个函数主要对数据的维度进行压缩 去掉维数为1的的维度 比如是一行或者一列这种 一个一行三列 1 3 的数去掉第一个维数为一的维度之后就变成 3 行 torch squeeze a 就
  • 项目答辩PPT(一)

    以 运动APP 为例展开 1 项目整体介绍 定位和需求 在银行工作的王小姐是个健身爱好者 从制订健身计划 到约人一起跑步 再到周末预订场地和家人打一场羽毛球赛 她用一部手机轻松搞定 现在有了手机APP 运动方便多了 王小姐的运动方式是如今都
  • QT创建右键快捷菜单

    0 目标 在Qcommbobox右键出来菜单 点击BCC校验 自动算出校验值填入编辑框 1 UI界面选择Action editor 新建action 记住对象名 actionBCC 右键action 点击转到槽 选择triggered 点击
  • 指针(一)——指针与二级指针

    一 指针理解 指针是一个变量 用来存放地址的变量 指针变量 是一个变量 是指有一个存储空间 里面放的是指针 变量指针 变量的地址 指针的存在是为了方便计算机的内存管理 经过计算和权衡 我们发现 一个字节给一个地址是比较合适的 在32位的机器
  • 给wangeditor添加上标、下标功能

    我使用的wangeditor没有上标和下标功能 以下是自己添加功能的方法 1 设计功能的函数和原型 Sup menu 构造函数 function Sup editor this editor editor this elem div cla
  • 高性能MySQL实战(一):表结构

    最近因需求改动新增了一些数据库表 但是在定义表结构时 具体列属性的选择有些不知其所以然 索引的添加也有遗漏和不规范的地方 所以我打算为创建一个高性能表的过程以实战的形式写一个专题 以此来学习和巩固这些知识 1 实战 我使用的 MySQL 版
  • 【深度学习】 Python 和 NumPy 系列教程(七):Python函数

    目录 一 前言 二 实验环境 三 Python函数基础 1 定义函数 2 参数传递 3 函数调用 4 返回值 5 函数文档字符串 四 将函数存储在模块中 1 创建模块 2 导入模块 a import 模块名 b from 模块名 impor