函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段。
函数能提高应用的模块性,和代码的重复利用率。你已经知道Python提供了许多内建函数,比如print()。但你也可以自己创建函数,这被叫做用户自定义函数。

1、语法

Python 定义函数使用 def 关键字,一般格式如下:

def 函数名(参数列表):
函数体

默认情况下,参数值和参数名称是按函数声明中定义的的顺序匹配起来的。

定义一个由自己想要功能的函数,以下是简单的规则:

  • 函数代码块以 def 关键词开头,后接函数标识符名称和圆括号 ()。
  • 任何传入参数和自变量必须放在圆括号中间,圆括号之间可以用于定义参数。
  • 函数的第一行语句可以选择性地使用文档字符串——用于存放函数说明。
  • 函数内容以冒号起始,并且缩进。
  • return [表达式] 结束函数,选择性地返回一个值给调用方。不带表达式的return相当于返回 None。

实例:

def hello():
return 'Hello world' def say(msg):
print(msg) def sun(num1, num2):
print('num1 + num2 =', num1, num2)

2、函数调用

Python内置了很多有用的函数,我们可以直接调用。要调用一个函数,需要知道函数的名称和参数,比如求绝对值的函数abs,只有一个参数:

print('100的绝对值:', abs(100))
print('-20的绝对值:', abs(-20))

以上代码,输出:

100的绝对值: 100
-20的绝对值: 20

定义一个函数:给了函数一个名称,指定了函数里包含的参数,和代码块结构。
这个函数的基本结构完成以后,你可以通过另一个函数调用执行,也可以直接从 Python 命令提示符执行。

3、参数传递 

在 python 中,类型属于对象,变量是没有类型的:

a=[1,2,3]

a="Python"

以上代码中,[1,2,3] 是 List 类型,"Python" 是 String 类型,而变量 a 是没有类型,它仅仅是一个对象的引用(一个指针),可以是 List 类型对象,也可以指向 String 类型对象。

python 函数的参数传递:

  • 不可变类型:类似 c++ 的值传递,如 整数、字符串、元组。如fun(a),传递的只是a的值,没有影响a对象本身。比如在 fun(a)内部修改 a 的值,只是修改另一个复制的对象,不会影响 a 本身。
  • 可变类型:类似 c++ 的引用传递,如 列表,字典。如 fun(la),则是将 la 真正的传过去,修改后fun外部的la也会受影响

python 中一切都是对象,严格意义我们不能说值传递还是引用传递,我们应该说传不可变对象和传可变对象。

什么是可更改(mutable)与不可更改(immutable)对象?

在 python 中,str, tuple, 和 number 是不可更改的对象,而 list,dict 等则是可以修改的对象。例如:

不可变类型:变量赋值 a=5 后再赋值 a=10,这里实际是新生成一个 int 值对象 10,再让 a 指向它,而 5 被丢弃,不是改变a的值,相当于新生成了a。

可变类型:变量赋值 la=[1,2,3,4] 后再赋值 la[2]=5 则是将 list la 的第三个元素值更改,本身la没有动,只是其内部的一部分值被修改了。

不可变对象在函数里修改了参数,原始的参数是不会改变的。例如:

def ChangeInt(a):
a = 10 b = 2
ChangeInt(b)
print(b) # 结果是 2

实例中有 int 对象 2,指向它的变量是 b,在传递给 ChangeInt 函数时,按传值的方式复制了变量 b,a 和 b 都指向了同一个 Int 对象,在 a=10 时,则新生成一个 int 值对象 10,并让 a 指向它。

可变对象在函数里修改了参数,那么原始的参数也被改变了。例如:

# 可写函数说明
def changeme(mylist):
"修改传入的列表"
mylist.append([1, 2, 3, 4])
print("函数内取值: ", mylist)
return # 调用changeme函数
mylist2 = [10, 20, 30]
changeme(mylist2)
print("函数外取值: ", mylist2)

传入函数的和在末尾添加新内容的对象用的是同一个引用。故输出结果如下:

函数内取值:  [10, 20, 30, [1, 2, 3, 4]]
函数外取值: [10, 20, 30, [1, 2, 3, 4]]

4、参数

  • 必需参数
  • 关键字参数
  • 默认参数
  • 不定长参数

什么是必需参数?必需参数须以正确的顺序传入函数。调用时的数量必须和声明时的一样。下面的实例中调用say()函数,你必须传入一个参数,不然会出现语法错误:

def say(msg):
print(msg) # 打印传入的参数 #调用say
say('hello world')

以上代码,输出:

hello world

但,我们如下调用时,就会报错:

def say(msg):
print(msg) # 打印传入的参数 #调用say
say()

以上代码,报错:

TypeError: say() missing 1 required positional argument: 'msg'

调用函数的时候,如果传入的参数数量不对,会报TypeError的错误,并且Python会明确地告诉你。

如果传入的参数数量是对的,但参数类型不能被函数所接受,也会报TypeError的错误,并且给出错误信息:str是错误的参数类型:

print(abs('abc'))

以上代码,会报错:

TypeError: bad operand type for abs(): 'str'

什么是关键字参数?关键字参数和函数调用关系紧密,函数调用使用关键字参数来确定传入的参数值。使用关键字参数允许函数调用时参数的顺序与声明时不一致,因为 Python 解释器能够用参数名匹配参数值。

我们来看实例:

#  申明一个函数say,第一个参数为:name,第二个参数为:msg
def say(name, msg):
print('%s 说:%s' %(name, msg)) # 打印传入的参数 # 普通方式调用say()
say('roy', 'hello world') # 用关键字参数的方式调用say()
say(msg='这是用关键字参数的方式调用的', name='roy')

以上代码,输出:

roy 说:hello world
roy 说:这是用关键字参数的方式调用的

我们可以看出,在第二次调用的时候,我们根本没有按照函数申明时的参数顺序去传递参数,但也是正确的输出了内容。

什么是默认参数?调用函数时,如果没有传递参数,则会使用默认参数。以下实例中如果没有传入 msg 参数,则使用默认值:

#  申明一个函数say,第一个参数为:name,第二个参数为:msg
def say(name, msg='hello world'):
print('%s 说:%s' %(name, msg)) # 打印传入的参数 # 调用
say('roy')

以上代码,输出:

roy 说:hello world

设置默认参数时,有几点要注意:

一是必选参数在前,默认参数在后,否则Python的解释器会报错(思考一下为什么默认参数不能放在必选参数前面);

二是如何设置默认参数。当函数有多个参数时,把变化大的参数放前面,变化小的参数放后面。变化小的参数就可以作为默认参数。

什么是不定长参数?你可能需要一个函数能处理比当初声明时更多的参数。这些参数叫做不定长参数,和上述几种参数不同,声明时不会命名。基本语法如下:

def functionname([formal_args,] *var_args_tuple ):
"函数_文档字符串"
function_suite
return [expression]

加了星号(*)的变量名会存放所有未命名的变量参数。如果在函数调用时没有指定参数,它就是一个空元组。我们也可以不向函数传递未命名的变量。如下实例:

# 可写函数说明
def printinfo(arg1, *vartuple):
"打印任何传入的参数"
print("输出: ")
print(arg1)
for var in vartuple:
print(var)
return # 调用printinfo 函数
printinfo(10)
printinfo(70, 60, 50)

以上代码,输出:

输出:
10
输出:
70
60
50

了解了这四种参数的形式了,我们来说下另一种参数传递的方式:分拆参数列表

例如,内建的range()函数期待单独的startstop参数。如果它们不能单独地获得,可以编写带有*操作的函数调用,来从一个列表或元组分拆出参数:

print(list(range(0, 3)))

args = [0, 3]
print(list(range(*args))) # range(*args) 等价于 range(0, 3)

以上代码,输出:

[0, 1, 2]
[0, 1, 2]

你会发现,range(*args) 等价于 range(0, 3),在上例中列表中的第一个元素被传递给了range()函数的第一个参数,第二个元素被传递给了函数的第二个参数。要注意的是,列表里的元素个数必须和函数必需参数的个数对应

同样的风格,字典可以通过**操作传递关键字参数:

def parrot(voltage, state='a stiff', action='voom'):
print("-- This parrot wouldn't", action, end=' ')
print("if you put", voltage, "volts through it.", end=' ')
print("E's", state, "!") d = {"voltage": "four million", "state": "bleedin' demised", "action": "VOOM"} parrot(**d)

以上代码,输出:

-- This parrot wouldn't VOOM if you put four million volts through it. E's bleedin' demised !

你会发现,字典的key会对照函数的参数,一一把key对应的value传入给函数

5、return语句

return [表达式] 语句用于退出函数,选择性地向调用方返回一个表达式。不带参数值的return语句返回None。

实例:

#  这是一个有return语句的函数
def hello():
return 'Hello world' # 这是一个有return语句的函数,默认返回None
def say(msg):
print(msg) # 调用 hello 并把它的结果赋值给 变量
msg = hello()
print(msg) # 调用 say
say('您好')

以上代码,输出:

Hello world
您好

6、注释

函数的第一行语句可以选择性地使用文档字符串——用于存放函数说明,例如:

def test():
"""
this is test
:return: int string
"""
return 1

可以调用 __doc__ 查看函数的文档:

def test():
"""
this is test
:return: int string
"""
return 1 print(test.__doc__)

以上代码,输出:

    this is test
:return: int string

在python3中你还可以这样为函数添加注释:

def dog(name:str, age:(1, 99), species:'狗狗的品种') -> tuple:
return (name, age, species)

如上,可以使用:对参数逐个进行注释,注释内容可以是任何形式,比如参数的类型、作用、取值范围等等,返回值使用->标注,所有的注释都会保存至函数的属性。

查看这些注释可以通过 __annotations__ 获取,结果会议字典的形式返回:

def dog(name:str, age:(1, 99), species:'狗狗的品种') -> tuple:
return (name, age, species) print(dog.__annotations__)

以上代码,输出:

{'name': <class 'str'>, 'age': (1, 99), 'species': '狗狗的品种', 'return': <class 'tuple'>}

另外,使用函数注释并不影响默认参数的使用:

def dog(name:str ='dobi', age:(1, 99) =3, species:'狗狗的品种' ='Labrador') -> tuple:
return (name, age, species) print(dog())

以上代码,输出:

('dobi', 3, 'Labrador')

python3 第十六章 - 函数的更多相关文章

  1. python3 第二十六章 - 内置函数之Number相关

    数学函数 函数 返回值 ( 描述 ) 实例 abs(x) 返回数字的绝对值,如abs(-10) 返回 10 print(abs(-10)) =======输出:====== 10 ceil(x) 返回 ...

  2. 《Python 学习手册4th》 第十六章 函数基础

    ''' 时间: 9月5日 - 9月30日 要求: 1. 书本内容总结归纳,整理在博客园笔记上传 2. 完成所有课后习题 注:“#” 后加的是备注内容 (每天看42页内容,可以保证月底看完此书) “重点 ...

  3. 《Linux命令行与shell脚本编程大全》 第十六章 学习笔记

    第十六章:创建函数 基本的脚本函数 创建函数 1.用function关键字,后面跟函数名 function name { commands } 2.函数名后面跟空圆括号,标明正在定义一个函数 name ...

  4. UNP学习笔记(第二十六章 线程)

    线程有时称为轻权进程(lightweight process) 同一进程内的所有线程共享相同的全局内存.这使得线程之间易于共享信息,然后这样也会带来同步的问题 同一进程内的所有线程处理共享全局变量外还 ...

  5. 进击的Python【第十六章】:Web前端基础之jQuery

    进击的Python[第十六章]:Web前端基础之jQuery 一.什么是 jQuery ? jQuery是一个JavaScript函数库. jQuery是一个轻量级的"写的少,做的多&quo ...

  6. Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第十六章:实例化和截头锥体裁切

    原文:Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第十六章:实例化和截头锥体裁切 代码工程地址: https://git ...

  7. 【C++】《C++ Primer 》第十六章

    第十六章 模板与泛型编程 面向对象编程和泛型编程都能处理在编写程序时不知道类型的情况. OOP能处理类型在程序允许之前都未知的情况. 泛型编程在编译时就可以获知类型. 一.定义模板 模板:模板是泛型编 ...

  8. Gradle 1.12 翻译——第十六章. 使用文件

    有关其它已翻译的章节请关注Github上的项目:https://github.com/msdx/gradledoc/tree/1.12,或訪问:http://gradledoc.qiniudn.com ...

  9. 第十六章——处理锁、阻塞和死锁(3)——使用SQLServer Profiler侦测死锁

    原文:第十六章--处理锁.阻塞和死锁(3)--使用SQLServer Profiler侦测死锁 前言: 作为DBA,可能经常会遇到有同事或者客户反映经常发生死锁,影响了系统的使用.此时,你需要尽快侦测 ...

随机推荐

  1. HDFS Architecture

    http://hadoop.apache.org/docs/r2.9.0/hadoop-project-dist/hadoop-hdfs/HdfsDesign.html Introduction Ha ...

  2. TPFrame框架之robot模块的基本使用

    经过几天的努力,基本功能版已经完成,有待大家验证... robot插件主要的功能是帮助我们写部分代码的基本,目前阶段已经实现后台curd数据管理的基本操作,更多功能待续... 1.首先下载robot插 ...

  3. linux交换空间

    swap空间有两种形式:一是交换分区,二是交换文件.总之对它的读写都是磁盘操作. linux内存通过 virtual memory 虚拟内存来管理整个内存, 虚拟内存管理着物理内存,也管理着swap交 ...

  4. Android基础_一次上传多张图片

    获取权限 <uses-permission android:name="android.permission.CAMERA"/> <uses-permission ...

  5. async/await 执行顺序详解

    随着async/await正式纳入ES7标准,越来越多的人开始研究据说是异步编程终级解决方案的 async/await.但是很多人对这个方法中内部怎么执行的还不是很了解,本文是我看了一遍技术博客理解 ...

  6. Java入门篇(一)——如何编写一个简单的Java程序

    最近准备花费很长一段时间写一些关于Java的从入门到进阶再到项目开发的教程,希望对初学Java的朋友们有所帮助,更快的融入Java的学习之中. 主要内容包括JavaSE.JavaEE的基础知识以及如何 ...

  7. python写一个翻译的小脚本

    起因: 想着上学看不懂English的PDF感慨万分........ 然后就有了翻译的脚本. 截图: 代码: #-*- coding:'utf-8' -*- import requests impor ...

  8. Git分支-分支简介

    源地址:https://git-scm.com/book/zh/ch3-1.html 几乎所有的版本控制系统都以某种形式支持分支. 使用分支意味着你可以把你的工作从开发主线上分离开来,以免影响开发主线 ...

  9. HDU 1847 Good Luck in CET-4 Everybody!(规律,博弈)

    Good Luck in CET-4 Everybody! Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K ...

  10. 二分查找c++简单模板

    //数组a[]中有n各元素,已经按升序排序,待查找的元素x sort(a,a+n); //升序排序 template<class Type> int BinarySearch(Type a ...