1.函数
函数是组织好的,可重复使用的,用来实现单一,或相关功能的代码段。
函数能够提高应用的模块性和代码的重复利用率,其中被用户自己创建的函数成为自定义函数。
定义一个自己想要功能的函数,规则如下:
- 函数代码块以def关键词开头,后接函数标识符名称和圆括号()。
- 任何传入参数和自变量必须放在圆括号中间,圆括号之间可以用于定义参数。
- 函数的第一行语句可以选择性的使用文档字符串-用于存放函数的说明。
- 函数内容以冒号起始,并且缩进。
- return[表达式]结束函数,选择性的返回一个值给调用方,不带表达式的return相当于返回None。
def functionname(形参):
"函数_文档字符串"
执行语句
return(expression)
默认情况下,参数值和参数名称是按函数声明中定义的顺序匹配起来的。
E.g:
def printme(str):
"打印传入的字符串到标准显示设备上"
print(str)
return
printme('Hello World!')
结果:
Hello World!
- 函数调用
定义一个函数只给了函数名称,指定了函数里包含的参数和代码块结构。
这个函数的基本结构完成以后,你可以通过另一个执行,也可以直接从Python提示符执行。
如下调用了printme()函数:
E.g:
def printme(str):
"打印任何传入的字符串"
print(str)
return
printme('我要调用用户自定义函数!')
printme('再一次调用同一函数!')
结果:
我要调用用户自定义函数!
再一次调用同一函数!
- 参数传递
在Python中,类型属于对象,变量是没有类型的。
a = [1,2,3]
a = 'I love China'
以上代码中,[1,2,3]是List类型,'I love China’是String类型,而变量a是没有类型的,它仅仅是一个对象的引用(一个指针),可以是List类型对象,也可以指向String类型对象。
可更改与不可更改对象
在Python中,string,tuples和numbers是不可更改对象,而list,dict等则是可以修改的对象。
不可变类型: 变量赋值a=5再赋值a=10,这里实际上是生成一个int值对象10,再让a指向它,而5被丢弃,不是改变a的值,相当于新生成了a。
可变类型: 变量赋值la=[1,2,3,4]后赋值la[2]=5则是将list la的第三个元素值更改,本身la没有动,只是其内部的一部分值被修改了。
Python函数的参数传递:
不可变类型: 类似C++的值传递,如整数,字符串,元组。如fun(a)传递的只是a的值,没有影响a对象本身。比如在fun(a)内部修改a的值,只是修改另一个复制的对象,不会影响a本身。
可变类型: 类似C++的引用传递,如列表,字典。如fun(la),则是将la真正的传递过去,修改后fun外部的la也会受影响。
传不可变对象实例:
E.g:
def ChangeInt(a):
a = 10
b = 2
ChangeInt(b)
print(b)
结果:
2
实例中有int对象2,指向它的变量是b,在传递给ChangeInt函数时,按传值的方式复制了变量b,a和b都指向了同一对象,在a=10时,则生成一个int值对象10,并让a指向它。
传可变对象实例:
E.g:
def changeme(mylist):
"修改传入的列表"
mylist.append([1,2,3,4])
print('函数内取值',mylist)
return
# 调用changeme函数
mylist = [10,20,30]
changeme(mylist)
print('函数外取值',mylist)
结果:
函数内取值 [10, 20, 30, [1, 2, 3, 4]]
函数外取值 [10, 20, 30, [1, 2, 3, 4]]
- 参数
必备参数: 必备参数须以正确的顺序传入函数,调用时的数量必须和声明时一样。
如下例调用printme()函数,必须传入一个参数,不然会出现语法错误:
E.g:
def printme(str):
"打印任何传入的字符串"
print(str)
return
# 调用printme函数
printme()
结果:
Traceback (most recent call last):
File "E:/pycharm/path/2019-3-28.py", line 39, in <module>
printme()
TypeError: printme() missing 1 required positional argument: 'str'
关键字参数: 关键字参数和函数调用关系密切,函数调用使用关键字来确定传入的参数值。使用关键字参数允许函数调用时参数的顺序和声明时不一致,因为Python解释器能够用参数名匹配参数值。
如下例在函数printme()调用时使用参数名:
E.g:
def printme(str):
"打印任何传入的字符串"
print(str)
return
# 调用printme函数
printme(str = 'My string')
结果:
My string
下例能将关键字参数顺序不重要展示得更清楚。
E.g:
def printinfo(name,age):
print('Name:',name)
print('Age:',age)
return
printinfo(age = 20 ,name = 'zcz')
结果:
Name: zcz
Age: 20
默认参数: 调用函数时,默认参数的值如果没有传入,则被认为是默认值。
如下例打印默认的age,如果age没有被传入:
E.g:
def printinfo(name , age = 21):
"打印任何传入的字符串"
print('Name:',name)
print('Age:',age)
return
printinfo(age = 20,name = 'zcz')
printinfo(name = 'zcz')
结果:
Name: zcz
Age: 20
Name: zcz
Age: 21
不定长参数: 可能需要一个函数能处理比当初声明时更多的参数,这些参数叫做不定长参数,和上述两种参数不同,声明是不会命名。
语法如下:
def functionname([formal_args],*var_args_tuple):
"函数_文档字符串"
执行语句
return [表达式]
加了星号(*)的变量会存放所有未命名的变量参数。
如下例:
E.g:
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
匿名函数: Python使用lambda来创建匿名函数。
- lambda只是一个表达式,函数体比def简单的多。
- lambda的主体是一个表达式,而不是一个代码块,仅仅能在lambda表达式中封装有限的逻辑进去。
- lambda函数拥有自己的命名空间,且不能访问自有参数列表之外或全局命名空间里的参数。
- 虽然lambda函数看起来只能写一行,却不等同于C或C++的内联函数,后者的目的是调用小函数时不占用栈内存从而增加运行效率。
语法:
lambda[arg1,[arg2,...,argn]]:表达式
如下例:
E.g:
sum = lambda arg1,arg2:arg1+arg2
print('相加后的值为:',sum(10,20))
print('相加后的值为:',sum(20,20))
结果:
相加后的值为: 30
相加后的值为: 40
return语句: return语句[表达式]退出函数,选择性的向调用方返回一个表达式,不带参数的return语句返回None。
E.g:
def sum(arg1,arg2):
total = arg1+arg2
print('函数内:',total)
return total
total = sum(10,20)
结果:
函数内: 30
全局变量和局部变量: 定义在函数内部的变量拥有一个局部作用域,定义在函数外的拥有全局作用域。
局部变量只能在其被声明的函数内部访问,而全局变量可以在整个程序范围内访问,调用函数时,所有在函数内声明的变量名称都将被加入到作用于中。
E.g:
total = 0
def sum(arg1,arg2):
total = arg1+arg2
print('函数内是局部变量:',total)
return total
sum(10,20)
print('函数外是全局变量:',total)
结果:
函数内是局部变量: 30
函数外是全局变量: 0
少年不被楼层误,余生不羁尽自由。
加油,加油!